Выбрать главу

Как видите, в этой программе не нужно создавать поток типа FileStream или освобождать его ресурсы. Все это делается в методе Сору() автоматически. Обратите также внимание на то, что в данной программе существующий файл не перезаписыва ется. Поэтому если целевой файл должен быть перезаписан, то для этой цели лучше воспользоваться второй из упоминавшихся ранее форм метода Сору(). Применение методов Exists() и GetLastAccessTime()

С помощью методов класса File очень легко получить нужные сведения о файле. Рассмотрим два таких метода: Exists() и GetLastAccessTime(). Метод Exists() определяет, существует ли файл, а метод GetLastAccessTime() возвращает дату и вре мя последнего доступа к файлу. Ниже приведены формы объявления обоих методов. static bool Exists(string путь) static DateTime GetLastAccessTime(string путь)

В обоих методах путь обозначает файл, сведения о котором требуется получить. Метод Exists() возвращает логическое значение true, если файл существует и до ступен для вызывающего процесса. А метод GetLastAccessTime() возвращает струк туру DateTime, содержащую дату и время последнего доступа к файлу. (Структура DateTime описывается далее в этой книге, но метод ToString() автоматически при водит дату и время к удобочитаемому виду.) С указанием недействительных аргумен тов или прав доступа при вызове обоих рассматриваемых здесь методов может быть связан целый ряд исключений, но в действительности генерируется только исключе ние IOException.

В приведенном ниже примере программы методы Exists() и GetLastAccessTime() демонстрируются в действии. В этой программе сначала определяется, существует ли файл под названием test.txt. Если он существует, то на экран выводит время послед него доступа к нему. // Применить методы Exists() и GetLastAccessTime(). using System; using System.IO; class ExistsDemo { static void Main() { if(File.Exists("test.txt")) Console.WriteLine("Файл существует. В последний раз он был доступен " + File.GetLastAccessTime("test.txt")); else Console.WriteLine("Файл не существует"); } }

Кроме того, время создания файла можно выяснить, вызвав метод GetCreationTime(), а время последней записи в файл, вызвав метод GetLastWriteTime(). Имеются так же варианты этих методов для представления данных о файле в формате всеобще го скоординированного времени (UTC). Попробуйте поэкспериментировать с ними. Преобразование числовых строк в их внутреннее представление

Прежде чем завершить обсуждение темы ввода-вывода, рассмотрим еще один спо соб, который может пригодиться при чтении числовых строк. Как вам должно быть уже известно, метод WriteLine() предоставляет удобные средства для вывода раз личных типов данных на консоль, включая и числовые значения встроенных типов, на пример int или double. При этом числовые значения автоматически преобразуются методом WriteLine() в удобную для чтения текстовую форму. В то же время ана логичный метод ввода для чтения и преобразования строк с числовыми значениями в двоичный формат их внутреннего представления не предоставляется. В частности, отсутствует вариант метода Read() специально для чтения строки "100", введенной с клавиатуры, и автоматического ее преобразования в соответствующее двоичное зна чение, которое может быть затем сохранено в переменной типа int. Поэтому данную задачу приходится решать другими способами. И самый простой из них — воспользо ваться методом Parse(), определенным для всех встроенных числовых типов данных.

Прежде всего необходимо отметить следующий важный факт: все встроенные в C# типы данных, например int или double, на самом деле являются не более чем псев донимами (т.е. другими именами) структур, определяемых в среде .NET Framework. В действительности тип в C# невозможно отличить от типа структуры в среде .NET Framework, поскольку один просто носит имя другого. В C# для поддержки значений простых типов используются структуры, и поэтому для типов этих значений имеются специально определенные члены структур.

Ниже приведены имена структур .NET и их эквиваленты в виде ключевых слов C# для числовых типов данных. Имя структуры в .NET Имя типа данных в C# Decimal decimal Double double Single float Int16 short Int32 int Int64 long UInt16 ushort UInt32 uint Uint64 ulong Byte byte Sbyte sbyte

Эти структуры определены в пространстве имен System. Следовательно, имя струк туры Int32 полностью определяется как System.Int32. Эти структуры предоставля ют обширный ряд методов, помогающих полностью интегрировать значения простых типов в иерархию объектов С#. А кроме того, в числовых структурах определяется ста тический метод Parse(), преобразующий числовую строку в соответствующий дво ичный эквивалент.

Существует несколько перегружаемых форм метода 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 не содержит допустимое число, определяемое вы зывающим типом данных. А если она содержит пустое значение, то генерируется ис ключение ArgumentNullException. Когда же значение в строке s превышает допу стимый диапазон чисел для вызывающего типа данных, то генерируется исключение OverflowException.

Методы синтаксического анализа позволяют без особого труда преобразовать чис ловое значение, введенное с клавиатуры или же считанное из текстового файла в виде строки, в соответствующий внутренний формат. В качестве примера ниже приведена программа, в которой усредняется ряд чисел, вводимых пользователем. Сначала поль зователю предлагается указать количество усредняемых значений, а затем это количе ство считывается методом 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.) В данной главе рассматриваются также анонимные методы, ковари антность, контравариантность и групповые преобразования методов. Делегаты