straustrup2 (852740), страница 69
Текст из файла (страница 69)
значение связанного потока перед обращением или 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)267Бьерн Страуструп.Язык программирования С++а не(##12),(##12)как можно было бы ожидать.
Однако, заметьте, что если бы влияние распространялось на все операциивывода чисел и строк, получился бы еще более неожиданный результат:(##12#),(##12#)С помощью стандартного манипулятора, показанного в 10.4.2.1, можно более элегантно задаватьразмера поля вывода.10.4.1.3 Состояние форматаВ классе ios содержится состояние формата, которое управляется функциями flags() и setf(). По сути этифункции нужны, чтобы установить или отменить следующие флаги:class ios {public:// управляющие форматом флаги:enum {skipws=01,left=02,right=04,internal=010,dec=020,oct=040,hex=0100,showbase=0200,showpoint=0400,uppercase=01000,showpos=02000,scientific=04000,fixed=010000,unitbuf=020000,stdio=040000//////////////////////////////////////пропуск обобщенных пробелов для inputполе выравнивания:добавление перед значениемдобавление после значениядобавление между знаком и значениемоснование целого:восьмеричноедесятичноешестнадцатеричноепоказать основание целоговыдать нули в конце'E', 'X' , а не 'e', 'x''+' для положительных чиселзапись числа типа float:.dddddd Edddddd.ddсброс в выходной поток:после каждой операциипосле каждого символа};//...};Смысл флагов будет разъяснен в последующих разделах.
Конкретные значения флагов зависят отреализации и даны здесь только для того, чтобы избежать синтаксически неверных конструкций.Определение интерфейса как набора флагов и операций для их установки или отмены - это оцененныйвременем, хотя и несколько устаревший прием. Основное его достоинство в том, что пользовательможет собрать воедино набор флагов, например, так:const int my_io_options =ios::left|ios::oct|ios::showpoint|ios::fixed;Такое множество флагов можно задавать как параметр одной операцииcout.flags(my_io_options);а также просто передавать между функциями одной программы:void your_function(int ios_options);void my_function()268Бьерн Страуструп.Язык программирования С++{// ...your_function(my_io_options);// ...}Множество флагов можно установить с помощью функции flags(), например:void your_function(int ios_options){int old_options = cout.flags(ios_options);// ...cout.flags(old_options); // reset options}Функция flags() возвращает старое значение множества флагов.
Это позволяет переустановитьзначения всех флагов, как показано выше, а также задать значение отдельному флагу. Например вызовmyostream.flags(myostream.flags()|ios::showpos);заставляет класс myostream выдавать положительные числа со знаком + и, в то же время, не меняетзначения других флагов. Получается старое значение множества флагов, к которому добавляется спомощью операции | флаг showpos.
Функция setf() делает то же самое, поэтому эквивалентная записьимеет видmyostream.setf(ios::showpos);После установки флаг сохраняет значение до явной отмены.Все-таки управление вводом-выводом с помощью установки и отмены флагов - грубое и ведущее кошибкам решение. Если только вы тщательно не изучите свое справочное руководство и не будетеприменять флаги только в простых случаях, как это делается в последующих разделах, то лучшеиспользовать манипуляторы (описанные в 10.4.2.1). Приемы работы с состоянием потока лучше изучитьна примере реализации класса, чем изучая интерфейс класса.10.4.1.4 Вывод целыхПрием задания нового значения множества флагов с помощью операции | и функций flags() и setf()работает только тогда, когда один бит определяет значение флага. Не такая ситуация при заданиисистемы счисления целых или вида выдачи вещественных.
Здесь значение, определяющее видвыдачи, нельзя задать одним битом или комбинацией отдельных битов.Решение, принятое в <iostream.h>, сводится к использованию версии функции setf(), работающей совторым "псевдопараметром", который показывает какой именно флаг мы хотим добавить к новомузначению.Поэтому обращенияcout.setf(ios::oct,ios::basefield); // восьмеричноеcout.setf(ios::dec,ios::basefield); // десятичноеcout.setf(ios::hex,ios::basefield); // шестнадцатеричноеустановят систему счисления, не затрагивая других компонентов состояния потока.
Если системасчисления установлена, она используется до явной переустановки, поэтомуcout << 1234 << ' ';cout << 1234 << ' ';cout.setf(ios::oct,ios::basefield);cout << 1234 << ' ';cout << 1234 << ' ';cout.setf(ios::hex,ios::basefield);cout << 1234 << ' ';cout << 1234 << ' ';// десятичное по умолчанию// восьмеричное// шестнадцатеричноенапечатает269Бьерн Страуструп.Язык программирования С++1234 1234 2322 2322 4d2 4d2Если появится необходимость указывать систему счисления для каждого выдаваемого числа, следуетустановить флаг showbase.
Поэтому, добавив перед приведенными выше обращениямиcout.setf(ios::showbase);мы получим1234 1234 02322 02322 0x4d2 0x4d2Стандартные манипуляторы, приведенные в $$10.4.2.1, предлагают более элегантный способопределения системы счисления при выводе целых.10.4.1.5 Выравнивание полейС помощью обращений к setf() можно управлять расположением символов в пределах поля:cout.setf(ios::left,ios::adjustfield);// влевоcout.setf(ios::right,ios::adjustfield);// вправоcout.setf(ios::internal,ios::adjustfield); // внутреннееБудет установлено выравнивание в поле вывода, определяемом функцией ios::width(), причем незатрагивая других компонентов состояния потока.Выравнивание можно задать следующим образом:cout.width(4);cout << '(' << -12 << ")\n";cout.width(4);cout.setf(ios::left,ios::adjustfield);cout << '(' << -12 << ")\n";cout.width(4);cout.setf(ios::internal,ios::adjustfield);cout << '(' << -12 << "\n";что выдаст( -12)(-12 )(- 12)Если установлен флаг выравнивания internal (внутренний), то символы добавляются между знаком ивеличиной.
Как видно, стандартным является выравнивание вправо.10.4.1.6 Вывод плавающих чисел.Вывод вещественных величин также управляется с помощью функций, работающих с состояниемпотока. В частности, обращения:cout.setf(ios::scientific,ios::floatfield);cout.setf(ios::fixed,ios::floatfield);cout.setf(0,ios::floatfield);// вернуться к стандартномуустановят вид печати вещественных чисел без изменения других компонентов состояния потока.Например:cout << 1234.56789 << '\n';cout.setf(ios::scientific,ios::floatfield);cout << 1234.56789 << '\n';cout.setf(ios::fixed,ios::floatfield);cout << 1234.56789 << '\n';напечатает1234.57270Бьерн Страуструп.Язык программирования С++1.234568e+031234.567890После точки печатается n цифр, как задается в обращенииcout.precision(n)По умолчанию n равно 6. Вызов функции precision влияет на все операции ввода-вывода свещественными до следующего обращения к precision, поэтомуcout.precision(8);cout << 1234.56789cout << 1234.56789cout.precision(4);cout << 1234.56789cout << 1234.56789<< '\n';<< '\n';<< '\n';<< '\n';выдаст1234.56791234.567912351235Заметьте, что происходит округление, а не отбрасывание дробной части.Стандартные манипуляторы, введенные в $$10.4.2.1, предлагают более элегантный способ заданияформата вывода вещественных.10.4.2 МанипуляторыК ним относятся разнообразные операции, которые приходится применять сразу перед или сразу послеоперации ввода-вывода.
Например:cout << x;cout.flush();cout << y;cin.eatwhite();cin >> x;Если писать отдельные операторы как выше, то логическая связь между операторами неочевидна, аесли утеряна логическая связь, программу труднее понять.Идея манипуляторов позволяет такие операции как flush() или eatwhite() прямо вставлять в списокопераций ввода-вывода. Рассмотрим операцию flush(). Можно определить класс с операциейoperator<<(), в котором вызывается flush():class Flushtype { };ostream& operator<<(ostream& os, Flushtype){return flush(os);}определить объект такого типаFlushtype FLUSH;и добиться выдачи буфера, включив FLUSH в список объектов, подлежащих выводу:cout << x << FLUSH << y << FLUSH ;Теперь установлена явная связь между операциями вывода и сбрасывания буфера.