ostringstream out_message;
out_message "ошибка: "
program_name "--" version
": " __FILE__ ": " __LINE__
" -- указатель равен 0; "
" а должен адресовать массив.\n";
// возвращаем строку, в которой находится сообщение
return out_message.str();
}
// ...
}
(В разделе 20.8 мы познакомимся со строковым вводом/выводом более подробно.)
Потоки ввода/вывода поддерживают два предопределенных типа: char и wchar_t. В этой главе мы расскажем только о чтении и записи в потоки данных типа char. Помимо них, в библиотеке iostream имеется набор классов и объектов для работы с типом wchar_t. Они отличаются от соответствующих классов, использующих тип char, наличием префикса ‘w’. Так, объект стандартного ввода называется wcin, стандартного вывода – wcout, стандартного вывода для ошибок – wcerr. Но набор заголовочных файлов для char и wchar_t один и тот же.
Классы для ввода/вывода данных типа wchar_t называются wostream, wistream, wiostream, для файлового ввода/вывода – wofstream, wifstream, wfstream, а для строкового – wostringstream, wistringstream, wstringstream.
20.1. Оператор вывода
Оператор вывода обычно применяется для записи на стандартный вывод cout. Например, программа
#include iostream
int main()
{
cout "сплетница Анна Ливия\n";
}
печатает на терминале строку:
сплетница Анна Ливия
Имеются операторы, принимающие аргументы любого встроенного типа данных, включая const char*, а также типов string и complex из стандартной библиотеки. Любое выражение, включая вызов функции, может быть аргументом оператора вывода при условии, что результатом его вычисления будет тип, принимаемый каким-либо вариантом этого оператора. Например, программа
#include iostream
#include string.h
int main()
{
cout "Длина 'Улисс' равна:\t";
cout strlen( "Улисс" );
cout '\n';
cout "Размер 'Улисс' равен:\t";
cout sizeof( "Улисс" );
cout endl;
}
выводит на терминал следующее:
Длина 'Улисс' равна:7
Размер 'Улисс' равен:8
endl – это манипулятор вывода, который вставляет в выходной поток символ перехода на новую строку, а затем сбрасывает буфер объекта ostream. (С буферизацией мы познакомимся в разделе 20.9.)
Операторы вывода, как правило, удобнее сцеплять в одну инструкцию. Например, предыдущую программу можно записать таким образом:
#include iostream
#include string.h
int main()
{
// операторы вывода можно сцеплять
cout "Длина 'Улисс' равна:\t";
strlen( "Улисс" ) '\n';
cout "Размер 'Улисс' равен:\t"
sizeof( "Улисс" ) endl;
}
Сцепление операторов вывода (и ввода тоже) возможно потому, что результатом выражения
cout "некоторая строка";
служит левый операнд оператора вывода, т.е. сам объект cout. Затем этот же объект передается следующему оператору и далее по цепочке (мы говорим, что оператор Имеется также предопределенный оператор вывода для указательных типов, который печатает адрес объекта. По умолчанию адреса отображаются в шестнадцатеричном виде. Например, программа
#include iostream
int main()
{
int i = 1024;
int *pi =
cout "i: " i
"\t&i:\t" &&&i&& '\n';
cout "*pi: " *pi
"\tpi:\t" pi endl
"\t\tπ:\t"&& π && endl;
}
выводит на терминал следующее:
i: 1024 &i: 0x7fff0b4
*pi: 1024 pi: 0x7fff0b4
π: 0x7fff0b0
Позже мы покажем, как напечатать адреса в десятичном виде.
Следующая программа ведет себя странно. Мы хотим напечатать адрес, хранящийся в переменной pstr:
#include iostream
const char *str = "vermeer";
int main()
{
const char *pstr = str;
cout "Адрес pstr равен: "
pstr endl;
}
Но после компиляции и запуска программа неожиданно выдает такую строку:
Адрес pstr равен: vermeer
Проблема в том, что тип const char* интерпретируется как C-строка. Чтобы все же напечатать адрес, хранящийся в pstr, необходимо подавить обработку типа const char* по умолчанию. Для этого мы сначала убираем спецификатор const, а затем приводим pstr к типу void*:
static_cast void*(const_cast char*(pstr))
Теперь программа выводит ожидаемый результат:
Адрес pstr равен: 0x116e8
А вот еще одна загадка. Нужно напечатать большее из двух чисел:
#include iostream