Г. Шилдт - Полный справочник по C# (1160789), страница 74
Текст из файла (страница 74)
Она записывает в файл данные различных типов, а затем считывает их.// Запись в файл двоичных данных с последующим//ихсчитыванием.using System;using System.10;class RWData {public static void Main() {BinaryWriter dataOut;BinaryReader dataln;int i = 10;double d = 1023.56;bool b = true;try {dataOut = newBinaryWriter(new FileStream("testdata",FileMode.Create));}catch(IOException exc) {Console.WriteLine(exc.Message +"\nHe удается открыть файл.
1 1 );396Часть I. Язык С#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);catch(IOException exc) {Console.WriteLine(exc.Message +"ХпОшибка при записи.");dataOut.Close();Console.WriteLine();// Теперь попробуем прочитать эти данные,try {dataln = newBinaryReader(new FileStream("testdata",FileMode.Open));}catch(FileNotFoundException exc) {Console.WriteLine(exc.Message +"\nHe удается открыть файл.");return;try {i = dataln.Readlnt32();Console.WriteLine("Считывание " + i ) ;d = dataln.ReadDouble();Console.WriteLine("Считывание " + d ) ;b = dataln.ReadBoolean();Console.WriteLine("Считывание " + b ) ;d = dataln.ReadDouble();Console.WriteLine("Считывание " + d ) ;}catch(IOException exc) {Console.WriteLine(exc.Message +"Ошибка при считывании.");dataln.Close ();Глава 14.
Использование средств ввода-вывода>397При выполнении этой профаммы были получены следующие результаты:Запись 10Запись 1023.56Запись TrueЗапись 90.28СчитываниеСчитываниеСчитываниеСчитывание101023.56True90.28»Если вы попробуете просмотреть содержимое файла t e s t d a t a , созданного этойпрофаммой, то увидите, что в нем содержатся двоичные данные, а не понятный длячеловека текст.А вот более практичный пример, который демонстрирует возможности С#-средствдвоичного ввода-вывода. Следующая профамма реализует очень простую профаммуинвентаризации. Для каждого элемента описи профамма хранит соответствующее наименование, имеющееся в наличии количество и стоимость. Профамма предлагаетпользователю ввести наименование элемента описи, а затем выполняет поиск в базеданных. Если элемент найден, на экране отображается соответствующая информация./* Использование классов BinaryReader и BinaryWriterдля реализации простой программы инвентаризации.
*/using System;using System.10;class Inventory {public static void Main() {BinaryWriter dataOut;BinaryReader dataln;string item; // Наименование элемента.int onhand; // Количество, имеющееся в наличии.double cost; // Цена.try {dataOut = newBinaryWriter(new FileStream("inventory.dat",FileMode.Create));}catch(IOException exc) {Console.WriteLine(exc.Message +"\nHe удается открыть файл.");return;}// Записываем некоторые инвентаризационные данные//вфайл,try {dataOut.Write("Молотки");dataOut.Write(10) ;dataOut.Write(3.95);dataOut.Write("Отвертки");dataOut.Write(18);dataOut.Write(1.50);398Часть I. Язык С #dataOut.Write("Плоскогубцы");dataOut.Write(5) ;dataOut.Write(4.95) ;dataOut.Write("Пилы");dataOut.Write(8) ;dataOut.Write(8.95) ;}catch(IOException exc) {Console.WriteLine(exc.Message +"\пОшибка при записи.");dataOut.Close() ;Console.WriteLine() ;// Теперь откроем файл инвентаризации// для чтения информации,try {dataln = newBinaryReader(new FileStream("inventory.dat",FileMode.Open));}catch(FileNotFoundException exc) {Console.WriteLine(exc.Message +"\nHe удается открыть файл.");return;// Поиск элемента, введенного пользователем.Console.Write("Введите наименование для поиска: " ) ;string what = Console.ReadLine();Console.WriteLine() ;try {for(;;) {// Считываем запись из базы данных,item = dataln.ReadString();onhand = dataln.Readlnt32();cost = dataln.ReadDouble();/* Если элемент в базе данных совпадает с элементомиз запроса, отображаем найденную информацию.
*/if(item.CompareTo(what) = = 0 ) {Console.WriteLine(item + ": " + onhand +" штук в наличии. " +"Цена: {0:С} за каждую единицу.",cost);Console.WriteLine("Общая стоимость по наименованию <{0}>: {1:С}." ,item, cost * onhand);break;catch(EndOfStreamException) {Console.WriteLine("Элемент не найден.");Глава 14. Использование средств ввода-вывода399catch(IOException exc) {Console.WriteLine(exc.Message + "Ошибка при чтении.");}dataln.Close();}}Вот результаты выполнения этой программы:Введите наименование для поиска: ОтверткиОтвертки: 18 штук в наличии. Цена: $1.50 за каждую единицу.Общая стоимость по наименованию <Отвертки>: $27.00.В этой программе стоит обратить внимание на то, как хранится информация о наличии товаров на складе, а именно — на двоичный формат хранения данных.
Следовательно, количество товаров, имеющихся в наличии, и их стоимость хранятся с использованием двоичного формата, а не в текстовом виде, удобном для восприятия человеком. Это позволяет выполнять вычисления над числовыми данными, не делаядополнительных преобразований.Хотелось бы также обратить ваше внимание на то, как обнаруживается здесь конецфайла. Поскольку при достижении конца потока методы ввода двоичной информациигенерируют исключение типа EndOfStreamException, эта программа просто считывает содержимое файла до тех пор, пока либо не найдет нужный элемент, либо несгенерируется это исключение.
Таким образом, для обнаружения конца файла в данном случае специального механизма не требуется.LJ Файлы с произвольным доступомДо сих пор мы использовали последовательные файлы, т.е. файлы, доступ к содержимому которых организован строго линейно, байт за байтом. Но в С# также возможен доступ к файлу, осуществляющийся случайным образом. В этом случае необходимо использовать метод Seek(), определенный в классе FileStream.
Этот метод позволяет установить индикатор позиции (или указатель позиции) в любое место файла.Заголовочная информация о методе Seek () имеет следующий вид:long Seek(long newPos, SeekOriginorigin)Здесь элемент newPos означает новую позицию, выраженную в байтах, файловогоуказателя относительно позиции, заданной элементом origin. Элемент origin может принимать одно из значений, определенных перечислением SeekOrigin.ЗначениеSeekOrigin. BeginОписаниеПоиск от начала файлаSeekOrigin.
CurrentПОИСК ОТ текущей ПОЗИЦИИSeekOrigin.EndПоиск от конца файлаПосле обращению к методу Seek () следующая операция чтения или записи данных будет выполняться на новой позиции в файле. Если при выполнении поиска возникнет какая-либо ошибка, генерируется исключение типа IOException. Если базовый поток не поддерживает функцию запроса нужной позиции, генерируется исключение типа NotSupportedException.400Часть I. Язык С#Рассмотрим пример, который демонстрирует выполнение операций ввода-вывода спроизвольным доступом. Следующая программа записывает в файл алфавит прописными буквами, а затем беспорядочно считывает его.// Демонстрация произвольного доступа к файлу.using System;using System.10;class RandomAccessDemo {public static void Main() {FileStream f;char ch;try {f = new FileStream("random.dat", FileMode.Create);}catch(IOException exc) {Console.WriteLine(exc.Message);return ;/ / Записываем в файл алфавит.f o r ( i n t i=0; i < 26; i++) {try {f.WriteByte((byte)('A'+i));}catch(IOException exc) {Console.WriteLine(exc.Message);return ;try {// Теперь считываем отдельные значения.f.Seek(O, SeekOrigin.Begin); // Поиск первого байта.ch = (char) f.ReadByte();Console.WriteLine("Первое значение равно " + ch) ;f.Seekd, SeekOrigin.Begin) ; // Поиск второго байта.ch = (char) f.ReadByte();Console.WriteLine("Второе значение равно " + ch) ;f.Seek(4, SeekOrigin.Begin); // Поиск пятого байта.
vch = (char) f.ReadByte();Console.WriteLine("Пятое значение равно " + ch);Console.WriteLine ();// Теперь считываем значения через одно.Console.WriteLine("Выборка значений через одно: " ) ;for(int i=0; i < 26; i += 2) {f.Seekd,SeekOrigin.Begin) ; / / Переход/ / к i-му байту.ch = (char) f.ReadByte();Console.Write(ch + " " ) ;Глава 14. Использование средств ввода-вывода401catch(IOException exc) {Console.WriteLine(exc.Message);Console.WriteLine();f.Close ();При выполнении этой программы получены такие результаты:Первое значение равно АВторое значение равно ВПятое значение равно ЕВыборка значений через одно:ACEGIKMOQSUWYИспользование класса Memory StreamНе всегда удобно выполнять операции ввода-вывода непосредственно с помощьюфизического устройства.
Иногда полезно считывать входные данные из массива илизаписывать их в массив. В этом случае стоит воспользоваться классом MemoryStream.Класс MemoryStream — это реализация класса Stream, в которой для операций ввода-вывода используются массивы байтов. Вот как выглядит конструктор этого класса:MemoryStream(byte[] buf)Здесь элемент buf— это массив байтов, который предполагается использовать воперациях ввода-вывода в качестве источника и/или приемника информации. В поток, создаваемый этим конструктором, можно записывать данные или считывать их внего. Этот поток поддерживает метод Seek (). Перед использованием этого конструктора необходимо позаботиться о достаточном размере массива buf, чтобы он позволилсохранить все направляемые в него данные.Вот пример программы, которая демонстрируетиспользование классаjyiemoryStream:// Демонстрация использования класса MemoryStream.using System;using System.10;class MemStrDemo {public static void Main() {byte[] storage = new byte[255];// Создаем поток с ориентацией на память.MemoryStream memstrm = new MemoryStream(storage) ;// Помещаем объект memstrm в оболочки StreamWriter// и StreamReader.StreamWriter memwtr = new StreamWriter(memstrm) ;StreamReader memrdr = new StreamReader(memstrm);// Записываем данные в память с помощью// объекта memwtr.for(int i=0; i < 10;402Часть I.