Класс BinaryReader служит оболочкой, в которую заключается байтовый поток,управляющий вводом двоичных данных. Ниже приведен наиболее часто употребляемый конструктор этого класса:BinaryReader(Stream input)
где input обозначает поток, из которого вводятся считываемые данные. Для чтенияиз входного файла в качестве параметра input может быть указан объект, создаваемый средствами класса FileStream. Если же поток, определяемый параметромinput, не был открыт для чтения данных или оказался недоступным по иным причинам, то генерируется исключение ArgumentException. По завершении ввода изпотока типа BinaryReader его нужно закрыть. При этом закрывается и базовыйпоток.
В классе BinaryReader определены методы, предназначенные для чтения данных всех встроенных в C# типов. Некоторые из этих методов перечислены в табл. 14.6.Следует, однако, иметь в виду, что в методе Readstring() считывается символьнаястрока, хранящаяся во внутреннем формате с указанием ее длины. Все методы данногокласса генерируют исключение IOException, если возникает ошибка ввода. Крометого, могут быть сгенерированы и другие исключения.
Таблица 14.6. Наиболее часто используемые методы, определенные в классе BinaryReaderМетодОписаниеbool ReadBoolean()Считывает значение логического типа boolbyte ReadByte()Считывает значение типа bytesbyte ReadSByte()Считывает значение типа sbytebyte[] ReadBytes(int count)Считывает количество count байтов и возвращает их в виде массиваchar ReadChar()Считывает значение типа charchar[] ReadChars(int count)Считывает количество count символов и возвращает их в виде массиваdecimal ReadDecimal()Считывает значение типа decimaldouble ReadDouble()Считывает значение типа doublefloat ReadSingle()Считывает значение типа floatshort ReadInt16()Считывает значение типа shortint ReadInt32()Считывает значение типа intlong ReadInt64()Считывает значение типа longushort ReadUIntl6()Считывает значение типа ushortuint ReadUInt32()Считывает значение типа uintulong ReadUInt64()Считывает значение типа ulongstring ReadString()Считывает значение типа string, представленное во внутреннем двоичном формате с указанием длины строки. Этот метод следует использовать для считывания строки, которая была записана средствами класса BinaryWriter
В классе BinaryWriter определены также три приведенных ниже варианта методаRead().
При неудачном исходе операции чтения эти методы генерируют исключениеIOException. Кроме того, в классе BinaryReader определен стандартный методClose().МетодОписаниеint Read()Возвращает целочисленное представление следующего доступного символа из вызывающего потока ввода. При обнаружении конца файла возвращает значение -1int Read(byte[] buffer, int offset, int count)Делает попытку прочитать количество count байтов в массив buffer, начиная с элемента buffer[offset], и возвращает количество успешно считанных байтовint Read(char[]buffer, int offset, int count)Делает попытку прочитать количество count символов в массив buffer, начиная с элемента buffer[offset], и возвращает количество успешно считанных символовДемонстрирование двоичного ввода-вывода
Ниже приведен пример программы, в котором демонстрируется применение классов BinaryReader и BinaryWriter для двоичного ввода-вывода. В этой программев файл записываются и считываются обратно данные самых разных типов.// Записать двоичные данные, а затем считать их обратно.using System;using System.IO;class RWData { static void Main() { BinaryWriter dataOut; BinaryReader dataIn; int i = 10; double d = 1023.56; bool b = true; string str = "Это тест"; // Открыть файл для вывода. try { dataOut = new BinaryWriter(new FileStream("testdata", FileMode.Create)); } catch(IOException exc) { Console.WriteLine("Ошибка открытия файла:\n" + exc.Message); return; } // Записать данные в файл. try { Console.WriteLine("Запись " + i); dataOut.Write(i); Console.WriteLine("Запись " + d); dataOut.Write(d); Console.WriteLine("Запись " + b); dataOut.Write(b); Console.WriteLine("Запись " + 12.2 * 7.4); dataOut.Write(12.2 * 7.4); Console.WriteLine("Запись " + str); dataOut.Write(str); } catch(IOException exc) { Console.WriteLine("Ошибка ввода-вывода:\n" + exc.Message); } finally { dataOut.Close(); } Console.WriteLine(); // А теперь прочитать данные из файла. try { dataIn = new BinaryReader(new FileStream("testdata", FileMode.Open)); } catch(IOException exc) { Console.WriteLine("Ошибка открытия файла:\n" + exc.Message); return; } try { i = dataIn.ReadInt32(); Console.WriteLine("Чтение " + i); d = dataIn.ReadDouble(); Console.WriteLine("Чтение " + d); b = dataIn.ReadBoolean () ; Console.WriteLine("Чтение " + b); d = dataIn.ReadDouble(); Console.WriteLine("Чтение " + d); str = dataIn.ReadString(); Console.WriteLine("Чтение " + str); } catch(IOException exc) { Console.WriteLine("Ошибка ввода-вывода:\n" + exc.Message); } finally { dataIn.Close(); } }}
Вот к какому результату приводит выполнение этой программы.Запись 10Запись 1023.56Запись TrueЗапись 90.28Запись Это тестЧтение 10Чтение 1023.56Чтение TrueЧтение 90.28Чтение Это тест
Если просмотреть содержимое файла testdata, который получается при выполнении этой программы, то можно обнаружить, что он содержит данные в двоичной,а не в удобочитаемой текстовой форме.
Далее следует более практический пример, демонстрирующий, насколько эффективным может быть двоичный ввод-вывод. Для учета каждого предмета хранения наскладе в приведенной ниже программе сначала запоминается наименование предмета, имеющееся в наличии, количество и стоимость, а затем пользователю предлагаетсяввести наименование предмета, чтобы найти его в базе данных. Если предмет найден,отображаются сведения о его запасах на складе./* Использовать классы BinaryReader и BinaryWriter дляреализации простой программы учета товарных запасов. */using System;using System.IO;class Inventory { static void Main() { BinaryWriter dataOut; BinaryReader detaIn; string item; // наименование предмета int onhand; // имеющееся в наличии количество double cost; // цена try { dataOut = new BinaryWriter(new FileStream("inventory.dat", FileMode.Create)); } catch(IOException exc) { Console.WriteLine("He удается открыть файл " + "товарных запасов для вывода"); Console.WriteLine("Причина: " + exc.Message); return; } // Записать данные о товарных запасах в файл. try { dataOut.Write("Молотки"); dataOut.Write(10); dataOut.Write(3.95); dataOut.Write("Отвертки"); dataOut.Write(18); dataOut.Write(1.50); dataOut.Write("Плоскогубцы"); dataOut.Write(5); dataOut.Write(4.95); dataOut.Write("Пилы"); dataOut.Write(8); dataOut.Write(8.95); } catch(IOException exc) { Console.WriteLine("Ошибка записи в файл товарных запасов"); Console.WriteLine("Причина: " + exc.Message); } finally { dataOut.Close(); } Console.WriteLine(); // А теперь открыть файл товарных запасов для чтения. try { dataIn = new BinaryReader(new FileStream("inventory.dat", FileMode.Open)); } catch(IOException exc) { Console.WriteLine("He удается открыть файл " + "товарных запасов для ввода"); Console.WriteLine("Причина: " + exc.Message); return; } // Найти предмет, введенный пользователем. Console.Write("Введите наименование для поиска: "); string what = Console.ReadLine(); Console.WriteLine(); try { for(;;) { // Читать данные о предмете хранения. item = dataIn.ReadString(); onhand = dataIn.ReadInt32(); cost = dataIn.ReadDouble(); // Проверить, совпадает ли он с запрашиваемым предметом. // Если совпадает, то отобразить сведения о нем. if(item.Equals(what, StringComparison.OrdinalIgnoreCase)) { Console.WriteLine(item + ": " + onhand + " штук в наличии. " + "Цена: {0:С} за штуку", cost); Console.WriteLine("Общая стоимость по наименованию <{0}>: {1:С}.", item, cost * onhand); break; } } } catch(EndOfStreamException) { Console.WriteLine("Предмет не найден."); } catch(IOException exc) { Console.WriteLine("Ошибка чтения из файла товарных запасов"); Console.WriteLine("Причина: " + exc.Message); } finally { dataIn.Glose(); } }}