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

class CheckoutRecord { // запись о выдаче

public:

// ...

private:

double book_id; // идентификатор книги

string title; // название

Date date_borrowed; // дата выдачи

Date date_due; // дата возврата

pairstring,string borrower; // кому выдана

vector pairstring,string wait_list; // очередь на книгу

};

20.5. Перегрузка оператора ввода

Перегрузка оператора ввода () похожа на перегрузку оператора вывода, но, к сожалению, возможностей для ошибок гораздо больше. Вот, например, его реализация для класса WordCount:

#include iostream

#include "WordCount.h"

/* необходимо модифицировать определение класса WordCount, чтобы

оператор ввода был другом

class WordCount {

friend ostream& operator&&( ostream&, const WordCount& );

friend istream& operator&&( istream&, const WordCount& );

*/

istream&

operator ( istream &is, WordCount &wd )

{

/* формат хранения объекта WordCount:

* 2 строка

* 7,3 12,36

*/

int ch;

/* прочитать знак ''. Если его нет,

* перевести поток в ошибочное состояние и выйти

*/

if ((ch = is.get()) != '' )

{

// is.setstate( ios_base::badbit );

return is;

}

// прочитать длину

int occurs;

is occurs;

// читать до обнаружения ; ошибки не контролируются

while ( is && (ch = is.get()) != '' ) ;

is wd._word;

// прочитать позиции вхождений;

// каждая позиция имеет формат: строка, колонка

for ( int ix = 0; ix occurs; ++ix )

{

int line, col;

// извлечь значения

while (is && (ch = is.get())!= '&' ) ;

is line;

while (is && (ch = is.get())!= ',' ) ;

is col;

while (is && (ch = is.get())!= '&' ) ;

wd._occurList.push_back( Location( line, col ));

}

return is;

}

* На этом примере показан целый ряд проблем, имеющих отношение к возможным ошибочным состояниям входного потока: поток, чтение из которого невозможно из-за неправильного формата, переводится в состояние faiclass="underline"

is.setstate( ios_base::failbit );

* операции вставки и извлечения из потока, находящегося в ошибочном состоянии, не работают:

while (( ch = is.get() ) != lbrace)

Инструкция зациклится, если объект istream будет находиться в ошибочном состоянии. Поэтому перед каждым обращением к get() проверяется отсутствие ошибки:

// проверить, находится ли поток "is" в "хорошем" состоянии

while ( is && ( ch = is.get() ) != lbrace)

Если объект istream не в "хорошем" состоянии, то его значение будет равно false. (О состояниях потока мы расскажем в разделе 20.7.)

Данная программа считывает объект класса WordCount, сохраненный оператором вывода из предыдущего раздела:

#include iostream

#include "WordCount.h"

int main()

{

WordCount readIn;

// operator( cin, readIn )

cin readIn;

if ( !cin ) {

cerr "Ошибка ввода WordCount" endl;

return -1;

}

// operator ( cout, readIn )

cout readIn endl;

}

Выводится следующее:

10 rosebud

11,3 11,8 14,2 34,6 49,7 67,5

81,2 82,3 91,4 97,8

Упражнение 20.9

Оператор ввода класса WordCount сам читает объекты класса Location. Вынесите этот код в отдельный оператор ввода класса Location.

Упражнение 20.10

Реализуйте оператор ввода для класса Date из упражнения 20.7 в разделе 20.4.

Упражнение 20.11

Реализуйте оператор ввода для класса CheckoutRecord из упражнения 20.8 в разделе 20.4.

20.6. Файловый ввод/вывод

Если программе необходимо работать с файлом, то следует включить в нее заголовочный файл fstream (который в свою очередь включает iostream):

#include fstream

Если файл будет использоваться только для вывода, мы определяем объект класса ofstream. Например:

ofstream outfile( "copy.out", ios::base::out );

Передаваемые конструктору аргументы задают имя открываемого файла и режим открытия. Файл типа ofstream может быть открыт либо – по умолчанию – в режиме вывода (ios_base::out), либо в режиме дозаписи (ios_base::app). Такое определение файла outfile2 эквивалентно приведенному выше:

// по умолчанию открывается в режиме вывода

ofstream outfile2( "copy.out" );

Если в режиме вывода открывается существующий файл, то все хранившиеся в нем данные пропадают. Если же мы хотим не заменить, а добавить данные, то следует открывать файл в режиме дозаписи: тогда новые данные помещаются в конец. Если указанный файл не существует, то он создается в любом режиме.