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" );
Если в режиме вывода открывается существующий файл, то все хранившиеся в нем данные пропадают. Если же мы хотим не заменить, а добавить данные, то следует открывать файл в режиме дозаписи: тогда новые данные помещаются в конец. Если указанный файл не существует, то он создается в любом режиме.