Бьерн Страуструп. Язык программирования С++


Ввод пользовательских типов



бет85/124
Дата16.07.2016
өлшемі3.27 Mb.
#204081
түріКнига
1   ...   81   82   83   84   85   86   87   88   ...   124

10.3.3 Ввод пользовательских типов


Операцию ввода для пользовательского типа можно определить в точности

так же, как и операцию вывода, но для операции ввода существенно, чтобы

второй параметр имел тип ссылки, например:


istream& operator>>(istream& s, complex& a)

/*

формат input рассчитан на complex; "f" обозначает float:



f

( f )


( f , f )

*/

{



double re = 0, im = 0;

char c = 0;


s >> c;

if (c == '(') {

s >> re >> c;

if (c == ',') s >> im >> c;

if (c != ')') s.clear(ios::badbit); // установим состояние

}

else {



s.putback(c);

s >> re;


}
if (s) a = complex(re,im);

return s;

}
Несмотря на сжатость кода, обрабатывающего ошибки, на самом деле

учитывается большая часть ошибок. Инициализация локальной переменной

с нужна для того, чтобы в нее не попало случайное значение, например

'(', в случае неудачной операции. Последняя проверка состояния потока

гарантирует, что параметр a получит значение только при успешном вводе.

Операция, устанавливающая состояние потока, названа clear()

(здесь clear - ясный, правильный),

поскольку чаще всего она используется для восстановления состояния потока

как good(); значением по умолчанию для параметра ios::clear() является

ios::goodbit.



10.4 Форматирование


Все примеры из 10.2 содержали неформатированный вывод, который являлся

преобразованием объекта в последовательность символов, задаваемую

стандартными правилами, длина которой также определяется этими

правилами. Часто программистам требуются более развитые возможности.

Так, возникает потребность контролировать размер памяти, необходимой

для операции вывода, и формат, используемый для выдачи чисел.

Точно так же допустимо управление некоторыми аспектами ввода.



10.4.1 Класс ios


Большинство средств управления вводом-выводом сосредоточены в классе

ios, который является базовым для ostream и istream. По сути здесь

находится управление связью между istream или ostream и буфером,

используемым для операций ввода-вывода. Именно класс ios контролирует:

как символы попадают в буфер и как они выбираются оттуда. Так, в классе

ios есть член, содержащий информацию об используемой при чтении или

записи целых чисел системы счисления (десятичная, восьмеричная или

шестнадцатеричная), о точности вещественных чисел и т.п., а также

функции для проверки и установки значений переменных, управляющих

потоком.
class ios {

//...


public:

ostream* tie(ostream* s); // связать input и output

ostream* tie(); // возвратить "tie"
int width(int w); // установить поле width

int width() const;


char fill(char); // установить символ заполнения

char fill() const; // вернуть символ заполнения


long flags(long f);

long flags() const;


long setf(long setbits, long field);

long setf(long);

long unsetf(long);
int precision(int); // установить точность для float

int precision() const;

int rdstate(); const; // состояния потоков, см. $$10.3.2

int eof() const;

int fail() const;

int bad() const;

int good() const;

void clear(int i=0);


//...

};
В 10.3.2 описаны функции, работающие с состоянием потока, остальные

приведены ниже.

10.4.1.1 Связывание потоков


Функция tie() может установить и разорвать связь между ostream и

istream. Рассмотрим пример:
main()

{

String s;



cout << "Password: ";

cin >> s;

// ...

}
Как можно гарантировать, что приглашение Password: появится на



экране прежде, чем выполниться операция чтения? Вывод в cout и ввод

из cin буферизуются, причем независимо, поэтому Password: появится

только по завершении программы, когда закроется буфер вывода.

Решение состоит в том, чтобы связать cout и cin с помощью

операции cin.tie(cout).

Если ostream связан с потоком istream, то буфер вывода выдается при

каждой операции ввода над istream. Тогда операции
cout << "Password: ";

cin >> s;


эквивалентны
cout << "Password: ";

cout.flush();

cin >> s;
Обращение is.tie(0) разрывает связь между потоком is и потоком, с

которым он был связан, если такой был. Подобно другим потоковым

функциям, устанавливающим определенное значение, tie(s) возвращает

предыдущее значение, т.е. значение связанного потока перед обращением

или 0. Вызов без параметра tie() возвращает текущее значение.

10.4.1.2 Поля вывода


Функция width() устанавливает минимальное число символов, использующееся

в последующей операции вывода числа или строки. Так в результате

следующих операций


cout.width(4);

cout << '(' << 12 << ')';


получим число 12 в поле размером 4 символа, т.е.
( 12)
Заполнение поля заданными символами или выравнивание можно установить с

помощью функции fill(), например:


cout.width(4);

cout.fill('#');

cout << '(' << "ab" << ')';
напечатает
(##ab)
По умолчанию поле заполняется пробелами, а размер поля по умолчанию

есть 0, что означает "столько символов, сколько нужно". Вернуть размеру

поля стандартное значение можно с помощью вызова
cout.width(0); // ``столько символов, сколько надо''
Функция width() задает минимальное число символов. Если появится больше

символов, они будут напечатаны все, поэтому


cout.width(4);

cout << '(' << "121212" << ")\n";


напечатает
(121212)
Причина, по которой разрешено переполнение поля, а не усечение вывода,

в том, чтобы избежать зависания при выводе. Лучше получить правильную

выдачу, выглядящую некрасиво, чем красивую выдачу, являющуюся

неправильной.

Вызов width() влияет только на одну следующую за ним операцию

вывода, поэтому


cout.width(4);

cout.fill('#');

cout << '(' << 12 << "),(" << '(' <<12 << ")\n";
напечатает
(##12),(12)
а не
(##12),(##12)
как можно было бы ожидать. Однако, заметьте, что если бы влияние

распространялось на все операции вывода чисел и строк, получился бы

еще более неожиданный результат:
(##12#),(##12#

)
С помощью стандартного манипулятора, показанного в 10.4.2.1, можно более

элегантно задавать размера поля вывода.



Достарыңызбен бөлісу:
1   ...   81   82   83   84   85   86   87   88   ...   124




©dereksiz.org 2024
әкімшілігінің қараңыз

    Басты бет