Полное руководство. С# 4.0
Шрифт:
Существует несколько перегружаемых форм метода Parse. Ниже приведены его простейшие варианты для каждой числовой структуры. Они выполняют преобразова ние с учетом местной специфики представления чисел. Следует иметь в виду, что каж дый метод возвращает двоичное значение, соответствующее преобразуемой строке. Структура Метод преобразования Decimal static decimal Parse(string s) Double static double Parse(string s) Single static float Parse(string s) Int64 static long Parse(string s) Int32 static int Parse(string s) Intl6 static short Parse(string s) UInt64 static ulong Parse(string s) UInt32 static uint Parse(string s) Ulnt16 static ushort Parse(string s) Byte static byte Parse(string s) Sbyte static sbyte Parse(string s)
Приведенные выше варианты метода Parse генерируют исключение FormatException, если строка s не содержит
Методы синтаксического анализа позволяют без особого труда преобразовать чис ловое значение, введенное с клавиатуры или же считанное из текстового файла в виде строки, в соответствующий внутренний формат. В качестве примера ниже приведена программа, в которой усредняется ряд чисел, вводимых пользователем. Сначала поль зователю предлагается указать количество усредняемых значений, а затем это количе ство считывается методом ReadLine и преобразуется из строки в целое число ме тодом Int32.Parse. Далее вводятся отдельные значения, преобразуемые методом Double.Parse из строки в их эквивалент типа double. // Эта программа усредняет ряд чисел, вводимых пользователем. using System; using System.IO; class AvgNums { static void Main { string str; int n; double sum = 0.0; double avg, t; Console.Write("Сколько чисел вы собираетесь ввести: "); str = Console.ReadLine; try { n = Int32.Parse(str); } catch(FormatException exc) { Console.WriteLine(exc.Message); return; } catch(OverflowException exc) { Console.WriteLine(exc.Message); return; } Console.WriteLine("Введите " + n + " чисел."); for(int i=0; i < n ; i++) { Console.Write(": "); str = Console.ReadLine; try { t = Double.Parse(str); } catch(FormatException exc) { Console.WriteLine(exc.Message); t = 0.0; } catch(OverflowException exc) { Console.WriteLine(exc.Message); t = 0; } sum += t; } avg = sum / n; Console.WriteLine("Среднее равно " + avg); } }
Выполнение этой программы может привести, например, к следующему резуль тату. Сколько чисел вы собираетесь ввести: 5 Введите 5 чисел. : 1.1 : 2.2 : 3.3 : 4.4 : 5.5 Среднее равно 3.3
Следует особо подчеркнуть, что для каждого преобразуемого значения необхо димо выбирать подходящий метод синтаксического анализа. Так, если попытаться преобразовать строку, содержащую значение с плавающей точкой, методом Int32. Parse, то искомый результат, т.е. числовое значение с плавающей точкой, получить не удастся.
Как пояснялось выше, при неудачном исходе преобразования метод Parse сге нерирует исключение. Для того чтобы избежать генерирования исключений при пре образовании числовых строк, можно воспользоваться методом TryParse, опреде ленным для всех числовых структур. В качестве примера ниже приведен один из вари антов метода TryParse, определяемых в структуре Int32: static bool TryParse(string s, out int результат)
где s обозначает числовую строку, передаваемую данному методу, который возвра щает соответствующий результат после преобразования с учетом выбираемой по умолчанию местной специфики представления чисел. (Конкретную местную специ фику представления чисел с учетом региональных стандартов можно указать в другом варианте данного метода.) При неудачном исходе преобразования, например, когда параметр s не содержит числовую строку в надлежащей форме, метод TryParse возвращает логическое значение false. В противном случае он возвращает логическое значение true. Следовательно, значение, возвращаемое этим методом, обязательно следует проверить, чтобы убедиться в удачном (или неудачном) исходе преобразова ния.
ГЛАВА 15. Делегаты, события и лямбда-выражения
В этой главе рассматриваются три новых средства С#: делегаты, события и лямбда-выражения. Делегат предоставляет возможность инкапсулировать метод, а событие уведомляет о том, что произошло некоторое действие. Делегаты и события тесно связаны друг с другом, поскольку событие основывается на делегате. Оба средства расширяют круг прикладных задача, решаемых при про граммировании на С#. А лямбда-выражение представляет собой новое синтаксическое средство, обеспечивающее упрощенный, но в то же время эффективный способ опре деления того, что по сути является единицей исполняемого кода. Лямбда-выражения обычно служат для работы с деле гатами и событиями, поскольку делегат может ссылаться на лямбда-выражение. (Кроме того, лямбда-выражения очень важны для языка LINQ, описываемого в главе 19.) В данной главе рассматриваются также анонимные методы, ковари антность, контравариантность и групповые преобразования методов. Делегаты
Начнем с определения понятия делегата. Попросту го воря, делегат представляет собой объект, который может ссылаться на метод. Следовательно, когда создается делегат, то в итоге получается объект, содержащий ссылку на метод. Более того, метод можно вызывать по этой ссылке. Иными словами, делегат позволяет вызывать метод, на который он ссылается. Ниже будет показано, насколько действенным оказывается такой принцип.
Следует особо подчеркнуть, что один и тот же делегат может быть использован для вызова разных методов во время выполнения программы, для чего достаточно изменить метод, на который ссылается делегат. Таким образом, метод, вызываемый делегатом, определяется во время выполнения, а не в процессе компиляции. В этом, собственно, и заключается главное преимущество делегата.
ПРИМЕЧАНИЕ Если у вас имеется опыт программирования на C/C++, то вам полезно будет знать, что делегат в C# подобен указателю на функцию в C/C++.
Тип делегата объявляется с помощью ключевого слова delegate. Ниже приведена общая форма объявления делегата: delegate возвращаемый_тип имя(список_параметров);
где возвращаемыйтип обозначает тип значения, возвращаемого методами, которые будут вызываться делегатом; имя — конкретное имя делегата; списокпараметров — параметры, необходимые для методов, вызываемых делегатом. Как только будет соз дан экземпляр делегата, он может вызывать и ссылаться на те методы, возвращаемый тип и параметры которых соответствуют указанным в объявлении делегата.
Самое главное, что делегат может служить для вызова любого метода с соответству ющей сигнатурой и возвращаемым типом. Более того, вызываемый метод может быть методом экземпляра, связанным с отдельным объектом, или же статическим методом, связанным с конкретным классом. Значение имеет лишь одно: возвращаемый тип и сигнатура метода должны быть согласованы с теми, которые указаны в объявлении делегата.
Для того чтобы показать делегат в действии, рассмотрим для начала простой при мер его применения. // Простой пример применения делегата. using System; // Объявить тип делегата. delegate string StrMod(string str); class DelegateTest { // Заменить пробелы дефисами. static string ReplaceSpaces(string s) { Console.WriteLine("Замена пробелов дефисами."); return s.Replace(' ', '-'); } // Удалить пробелы. static string RemoveSpaces(string s) { string temp = ""; int i; Console.WriteLine("Удаление пробелов."); for(i=0; i < s.Length; i++) if(s[i] != ' ') temp += s[i]; return temp; } // Обратить строку. static string Reverse(string s) { string temp = ""; int i, j; Console.WriteLine("Обращение строки."); for(j=0, i=s.Length-1; i >= 0; i--, j++) temp += s[i]; return temp; } static void Main { // Сконструировать делегат. StrMod strOp = new StrMod(ReplaceSpaces); string str; // Вызвать методы с помощью делегата. str = strOp("Это простой тест."); Console.WriteLine("Результирующая строка: " + str); Console.WriteLine; strOp = new StrMod(RemoveSpaces); str = strOp("Это простой тест."); Console.WriteLine("Результирующая строка: " + str); Console.WriteLine; strOp = new StrMod(Reverse); str = strOp("Это простой тест."); Console.WriteLine("Результирующая строка: " + str); } }
Вот к какому результату приводит выполнение этого кода. Замена пробелов дефисами. Результирующая строка: Это-простой-тест. Удаление пробелов. Результирующая строка: Этопростойтест. Обращение строки. Результирующая строка: .тсет йотсорп отЭ
Рассмотрим данный пример более подробно. В его коде сначала объявляется деле гат StrMod типа string, как показано ниже. delegate string StrMod(string str);
Как видите, делегат StrMod принимает один параметр типа string и возвращает одно значение того же типа.
Далее в классе DelegateTest объявляются три статических метода с одним пара метром типа string и возвращаемым значением того же типа. Следовательно, они со ответствуют делегату StrMod. Эти методы видоизменяют строку в той или иной фор ме. Обратите внимание на то, что в методе ReplaceSpaces для замены пробелов дефисами используется один из методов типа string — Replace.
В методе Main создается переменная экземпляра strOp ссылочного типа StrMod и затем ей присваивается ссылка на метод ReplaceSpaces. Обратите особое внима ние на следующую строку кода. StrMod strOp = new StrMod(ReplaceSpaces);