23.3 – Вывод данных с помощью ostream и ios
В этом разделе мы рассмотрим различные аспекты класса библиотеки iostream
для вывода данных (ostream
).
Примечание. Все функции ввода/вывода в этом уроке находятся в пространстве имен std
. Это означает, что все объекты и функции ввода/вывода должны иметь префикс std::
, или должна быть использована инструкция using namespace std;
.
Оператор вставки
Оператор вставки используется для записи данных в выходной поток. В C++ уже есть предопределенные операции вставки для всех встроенных типов данных, а как перегрузить оператор вставки для пользовательских классов, мы рассмотрели в уроке «13.4 – Перегрузка операторов ввода/вывода».
В уроке о потоках вы видели, что и istream
, и ostream
были производными от класса с именем ios
. Одна из задач ios
(и ios_base
) – управлять параметрами форматирования вывода.
Форматирование
Есть два способа изменить параметры форматирования: флаги и манипуляторы. Вы можете думать о флагах как о логических переменных, которые можно включать и выключать. Манипуляторы – это объекты, помещенные в поток, которые влияют на способ ввода и вывода.
Чтобы включить флаг, используйте функцию setf()
с соответствующим флагом в качестве параметра. Например, по умолчанию C++ не выводит знак + перед положительными числами. Однако, используя флаг std::ios::showpos
, мы можем изменить это поведение:
std::cout.setf(std::ios::showpos); // включить флаг std::ios::showpos
std::cout << 27 << '\n';
Это приводит к следующему выводу:
+27
Можно включить сразу несколько флагов ios
с помощью оператора ИЛИ (|
):
// включить флаги std::ios::showpos и std::ios::uppercase
std::cout.setf(std::ios::showpos | std::ios::uppercase);
std::cout << 27 << '\n';
Чтобы отключить флаг, используйте функцию unsetf()
:
std::cout.setf(std::ios::showpos); // включить флаг std::ios::showpos
std::cout << 27 << '\n';
std::cout.unsetf(std::ios::showpos); // выключить флаг std::ios::showpos
std::cout << 28 << '\n';
Это приводит к следующему выводу:
+27
28
Есть еще одна хитрость при использовании setf()
, о которой следует упомянуть. Многие флаги принадлежат группам, называемым группами форматирования. Группа форматирования – это группа флагов, которые управляют аналогичными (иногда взаимоисключающими) параметрами форматирования. Например, группа форматирования с именем basefield
содержит флаги oct
, dec
и hex
, которые управляют основанием целочисленных значений. По умолчанию установлен флаг dec
. Следовательно, если мы сделаем это:
std::cout.setf(std::ios::hex); // попытаться включить шестнадцатеричный вывод
std::cout << 27 << '\n';
То получим такой вывод:
27
Не сработало! Причина в том, что setf()
только включает флаги – этого недостаточно, чтобы отключить взаимоисключающие флаги. Следовательно, когда мы включили std::hex
, std::ios::dec
всё еще был включен, а std::ios::dec
явно имеет приоритет. Есть два способа обойти эту проблему.
Во-первых, мы можем отключить std::ios::dec
, чтобы был установлен только std::hex
:
std::cout.unsetf(std::ios::dec); // выключаем десятичный вывод
std::cout.setf(std::ios::hex); // включаем шестнадцатеричный вывод
std::cout << 27 << '\n';
Теперь мы получаем ожидаемый результат:
1b
Второй способ – использовать другую форму setf()
, которая принимает два параметра: первый параметр – это флаг, который нужно установить, а второй – это группа форматирования, к которой он принадлежит. При использовании этой формы setf()
все флаги, принадлежащие группе, отключаются, и включается только переданный флаг. Например:
// Включаем std::ios::hex как единственный флаг std::ios::basefield
std::cout.setf(std::ios::hex, std::ios::basefield);
std::cout << 27 << '\n';
Это также дает ожидаемый результат:
1b
Использование setf()
и unsetf()
может быть неудобным, поэтому C++ предоставляет второй способ изменения параметров форматирования: манипуляторы. В манипуляторах хорошо то, что они достаточно умны, чтобы включать и выключать соответствующие флаги. Вот пример использования некоторых манипуляторов для изменения основания чисел:
std::cout << std::hex << 27 << '\n'; // выводим 27 в шестнадцатеричном формате
std::cout << 28 << '\n'; // мы все еще в шестнадцатеричном формате
std::cout << std::dec << 29 << '\n'; // вернуться к десятичной системе счисления
Эта программа создает следующий вывод:
1b
1c
29
В общем, пользоваться манипуляторами намного проще, чем устанавливать и сбрасывать флаги. Многие параметры доступны как через флаги, так и через манипуляторы (например, изменение основания числа), однако некоторые параметры доступны только через флаги или только через манипуляторы, поэтому важно знать, как использовать и то, и другое.
Полезные инструменты форматирования
Вот список некоторых наиболее полезных флагов, манипуляторов и функций-членов. Флаги находятся в классе std::ios
, манипуляторы находятся в пространстве имен std
, а функции-члены живут в классе std::ostream
.
Группа | Флаг | Назначение |
---|---|---|
std::ios::boolalpha | Если установлен, логические значения выводят как "true" или "false". Если не установлен, логические значения выводят как 0 или 1. |
Манипулятор | Назначение |
---|---|
std::boolalpha | Логические значения выводят как "true" или "false". |
std::noboolalpha | Логические значения выводят как 0 или 1 (по умолчанию). |
Пример:
std::cout << true << " " << false << '\n';
std::cout.setf(std::ios::boolalpha);
std::cout << true << " " << false << '\n';
std::cout << std::noboolalpha << true << " " << false << '\n';
std::cout << std::boolalpha << true << " " << false << '\n';
Результат:
1 0
true false
1 0
true false
Группа | Флаг | Назначение |
---|---|---|
std::ios::showpos | Если установлен, перед положительными числами ставится +. |
Манипулятор | Назначение |
---|---|
std::showpos | Добавляет к положительным числам префикс + |
std::noshowpos | Не ставит перед положительными числами знак + |
Пример:
std::cout << 5 << '\n';
std::cout.setf(std::ios::showpos);
std::cout << 5 << '\n';
std::cout << std::noshowpos << 5 << '\n';
std::cout << std::showpos << 5 << '\n';
Результат:
5
+5
5
+5
Группа | Флаг | Назначение |
---|---|---|
std::ios::uppercase | Если установлен, используются буквы верхнего регистра. |
Манипулятор | Назначение |
---|---|
std::uppercase | Использует буквы верхнего регистра |
std::nouppercase | Использует буквы нижнего регистра |
Пример:
std::cout << 12345678.9 << '\n';
std::cout.setf(std::ios::uppercase);
std::cout << 12345678.9 << '\n';
std::cout << std::nouppercase << 12345678.9 << '\n';
std::cout << std::uppercase << 12345678.9 << '\n';
Результат:
1.23457e+007
1.23457E+007
1.23457e+007
1.23457E+007
Группа | Флаг | Назначение |
---|---|---|
std::ios::basefield | std::ios::dec | Печатает значения в десятичном формате (по умолчанию) |
std::ios::basefield | std::ios::hex | Печатает значения в шестнадцатеричном формате |
std::ios::basefield | std::ios::oct | Печатает значения в восьмеричном формате |
std::ios::basefield | (нет) | Печатает значения в соответствии с начальными символами значения |
Манипулятор | Назначение |
---|---|
std::dec | Печатает значения в десятичном формате |
std::hex | Печатает значения в шестнадцатеричном формате |
std::oct | Печатает значения в восьмеричном формате |
Пример:
std::cout << 27 << '\n';
std::cout.setf(std::ios::dec, std::ios::basefield);
std::cout << 27 << '\n';
std::cout.setf(std::ios::oct, std::ios::basefield);
std::cout << 27 << '\n';
std::cout.setf(std::ios::hex, std::ios::basefield);
std::cout << 27 << '\n';
std::cout << std::dec << 27 << '\n';
std::cout << std::oct << 27 << '\n';
std::cout << std::hex << 27 << '\n';
Результат:
27
27
33
1b
27
33
1b
К этому моменту вы должны увидеть взаимосвязь между настройкой форматирования с помощью флагов и с помощью манипуляторов. В будущих примерах мы будем использовать манипуляторы, если они доступны.
Точность, формат записи и десятичные точки
Используя манипуляторы (или флаги), можно изменить точность и формат отображения чисел с плавающей запятой. Есть несколько вариантов форматирования, которые комбинируются сложным образом, поэтому мы рассмотрим их подробнее.
Группа | Флаг | Назначение |
---|---|---|
std::ios::floatfield | std::ios::fixed | Использует десятичную запись для чисел с плавающей запятой |
std::ios::floatfield | std::ios::scientific | Использует экспоненциальную запись для чисел с плавающей запятой |
std::ios::floatfield | (нет) | Использует десятичную запись для чисел с несколькими цифрами, в противном случае – экспоненциальную запись |
std::ios::floatfield | std::ios::showpoint | Всегда показывать десятичную точку и завершающие нули для значений с плавающей запятой |
Манипулятор | Назначение |
---|---|
std::fixed | Использует десятичную запись значений |
std::scientific | Использует экспоненциальную запись значений |
std::showpoint | Показывает десятичную точку и завершающие нули для значений с плавающей запятой |
std::noshowpoint | Не показывать десятичную точку и завершающие нули для значений с плавающей запятой |
std::setprecision(int) | Устанавливает точность чисел с плавающей запятой (определен в iomanip.h) |
Функция-член | Назначение |
---|---|
std::precision() | Возвращает текущую точность чисел с плавающей запятой |
std::precision(int) | Устанавливает точность чисел с плавающей запятой и возвращает старую точность |
Если используется фиксированная или экспоненциальная запись, точность определяет, сколько знаков отображается в дробной части. Обратите внимание, что если точность меньше количества значащих цифр, число будет округлено.
std::cout << std::fixed << '\n';
std::cout << std::setprecision(3) << 123.456 << '\n';
std::cout << std::setprecision(4) << 123.456 << '\n';
std::cout << std::setprecision(5) << 123.456 << '\n';
std::cout << std::setprecision(6) << 123.456 << '\n';
std::cout << std::setprecision(7) << 123.456 << '\n';
std::cout << std::scientific << '\n';
std::cout << std::setprecision(3) << 123.456 << '\n';
std::cout << std::setprecision(4) << 123.456 << '\n';
std::cout << std::setprecision(5) << 123.456 << '\n';
std::cout << std::setprecision(6) << 123.456 << '\n';
std::cout << std::setprecision(7) << 123.456 << '\n';
Дает следующий результат:
123.456
123.4560
123.45600
123.456000
123.4560000
1.235e+002
1.2346e+002
1.23456e+002
1.234560e+002
1.2345600e+002
Если не используется ни фиксированная, ни экспоненциальная запись, точность определяет, сколько следует отображать значащих цифр. Опять же, если точность меньше количества значащих цифр, число будет округлено.
std::cout << std::setprecision(3) << 123.456 << '\n';
std::cout << std::setprecision(4) << 123.456 << '\n';
std::cout << std::setprecision(5) << 123.456 << '\n';
std::cout << std::setprecision(6) << 123.456 << '\n';
std::cout << std::setprecision(7) << 123.456 << '\n';
Дает следующий результат:
123
123.5
123.46
123.456
123.456
Используя манипулятор или флаг showpoint
, вы можете заставить поток записывать десятичную точку и завершающие нули.
std::cout << std::showpoint << '\n';
std::cout << std::setprecision(3) << 123.456 << '\n';
std::cout << std::setprecision(4) << 123.456 << '\n';
std::cout << std::setprecision(5) << 123.456 << '\n';
std::cout << std::setprecision(6) << 123.456 << '\n';
std::cout << std::setprecision(7) << 123.456 << '\n';
Дает следующий результат:
123.
123.5
123.46
123.456
123.4560
Вот сводная таблица с еще несколькими примерами:
Тип записи | Точность | 12345.0 | 0.12345 |
---|---|---|---|
Обычный | 3 | 1.23e+004 | 0.123 |
4 | 1.235e+004 | 0.1235 | |
5 | 12345 | 0.12345 | |
6 | 12345 | 0.12345 | |
Показывать десятичную точку | 3 | 1.23e+004 | 0.123 |
4 | 1.235e+004 | 0.1235 | |
5 | 12345. | 0.12345 | |
6 | 12345.0 | 0.123450 | |
Фиксированная запись | 3 | 12345.000 | 0.123 |
4 | 12345.0000 | 0.1235 | |
5 | 12345.00000 | 0.12345 | |
6 | 12345.000000 | 0.123450 | |
Экспоненциальная запись | 3 | 1.235e+004 | 1.235e-001 |
4 | 1.2345e+004 | 1.2345e-001 | |
5 | 1.23450e+004 | 1.23450e-001 | |
6 | 1.234500e+004 | 1.234500e-001 |
Ширина, символы заполнения и выравнивание
Обычно при печати чисел числа печатаются без учета пространства вокруг них. Однако печать чисел можно выровнять по левому или правому краю. Для этого мы должны сначала определить ширину поля, которая определяет количество выходных мест печати символов, которые будут заниматься печатаемым значением. Если фактическое напечатанное число меньше ширины поля, оно будет выровнено по левому или правому краю (как указано). Если фактическое число больше ширины поля, оно не будет усечено – оно приведет к переполнению поля.
Группа | Флаг | Назначение |
---|---|---|
std::ios::adjustfield | std::ios::internal | Выравнивает знак числа по левому краю, а значение – по правому краю |
std::ios::adjustfield | std::ios::left | Выравнивание знака и значения по левому краю |
std::ios::adjustfield | std::ios::right | Выравнивает знак и значение по правому краю (по умолчанию) |
Манипулятор | Назначение |
---|---|
std::internal | Выравнивает знак числа по левому краю, а значение – по правому краю |
std::left | Выравнивание знака и значения по левому краю |
std::right | Выравнивает знак и значение по правому краю |
std::setfill(char) | Устанавливает параметр как символ заполнения (определен в iomanip.h) |
std::setw(int) | Устанавливает ширину поля для ввода и вывода в значение параметра (определен в iomanip.h) |
Функция-член | Назначение |
---|---|
std::fill() | Возвращает текущий символ заполнения |
std::fill(char) | Устанавливает символ заполнения и возвращает старый символ заполнения |
std::width() | Возвращает текущую ширину поля |
std::width(int) | Устанавливает новую ширину поля и возвращает старую ширину поля |
Чтобы использовать любое из этих средств форматирования, мы сначала должны установить ширину поля. Это можно сделать с помощью функции-члена width(int)
или манипулятора setw()
. Обратите внимание, что по умолчанию используется выравнивание по правому краю.
// выводим значение по умолчанию без ширины поля
std::cout << -12345 << '\n';
// выводим значение по умолчанию с шириной поля
std::cout << std::setw(10) << -12345 << '\n';
// печать с выравниванием по левому краю
std::cout << std::setw(10) << std::left << -12345 << '\n';
// печать с выравниванием по правому краю
std::cout << std::setw(10) << std::right << -12345 << '\n';
// печать с внутренним выравниванием
std::cout << std::setw(10) << std::internal << -12345 << '\n';
Это дает следующий результат:
-12345
-12345
-12345
-12345
- 12345
Следует отметить, что setw()
и width()
влияют только на следующую инструкцию вывода. Они не постоянны, как некоторые другие флаги/манипуляторы.
Теперь давайте установим символ заполнения и выполним тот же пример:
std::cout.fill('*');
// выводим значение по умолчанию без ширины поля
std::cout << -12345 << '\n';
// выводим значение по умолчанию с шириной поля
std::cout << std::setw(10) << -12345 << '\n';
// печать с выравниванием по левому краю
std::cout << std::setw(10) << std::left << -12345 << '\n';
// печать с выравниванием по правому краю
std::cout << std::setw(10) << std::right << -12345 << '\n';
// печать с внутренним выравниванием
std::cout << std::setw(10) << std::internal << -12345 << '\n';
Это дает следующий результат:
-12345
****-12345
-12345****
****-12345
-****12345
Обратите внимание, что все пустые места в поле заполнены символом заполнения.
Класс ostream
и библиотека iostream
содержат функции вывода, флаги и манипуляторы, которые могут быть полезны в зависимости от того, что вам нужно сделать. Как и в случае с классом istream
, эти темы больше подходят для учебника или книги, посвященной стандартной библиотеке.