Обратите внимание на то, что ширина поля не является инертным параметром. Во всех трех случаях первое и последнее числа по умолчанию выведены с максимальным количеством цифр, которые допускает текущий формат. Иначе говоря, если мы непосредственно перед выводом не укажем ширину поля, то понятие поля вообще не будет использовано.
ПОПРОБУЙТЕ
Создайте простую таблицу, содержащую фамилию, имя, номер телефона и адрес электронной почты не менее пяти ваших друзей. Поэкспериментируйте с разной шириной поля, пока не найдете приемлемый вид таблицы.
11.3. Открытие файла и позиционирование
В языке С++ файл — это абстракция возможностей операционной системы. Как указано в разделе 10.3, файл — это последовательность байтов, пронумерованных начиная с нуля.
Вопрос заключается лишь в том, как получить доступ к этим байтам. При работе с потоками iostream
вид доступа определяется в тот момент, когда мы открываем файл и связываем с ним поток. Поток сам определяет, какие операции можно выполнить после открытия файла и каков их смысл. Например, когда мы открываем для файла поток istream
, то можем прочитать его содержимое, а когда открываем для файла поток ostream
, то можем записать в него данные.
11.3.1. Режимы открытия файлов
Файл можно открыть в одном из нескольких режимов. По умолчанию поток ifstream
открывает файлы для чтения, а поток ofstream
— для записи. Эти операции удовлетворяют большинство наших потребностей. Однако существует несколько альтернатив.
Режим открытия файла можно указать после его имени. Рассмотрим пример.
ofstream of1(name1); // по умолчанию ios_base::out
ifstream if1(name2); // по умолчанию ios_base::in
ofstream ofs(name, ios_base::app); // по умолчанию ofstream —
// для записи
fstream fs("myfile", ios_base::in|ios_base::out); // для ввода и вывода
Символ |
в последнем примере — это побитовый оператор ИЛИ (раздел A.5.5), который можно использовать для объединения режимов. Опция app
часто используется для записи регистрационных файлов, в которых записи всегда добавляются в конец.
В любом случае конкретный режим открытия файла может зависеть от операционной системы. Если операционная система не может открыть файл в требуемом режиме, то поток перейдет в неправильное состояние.
if (!fs) // Ой: мы не можем открыть файл в таком режиме
В большинстве ситуаций причиной сбоя при открытии файла для чтения является его отсутствие.
ifstream ifs("redungs");
if (!ifs) // ошибка: невозможно открыть файл readings для чтения
В данном случае причиной ошибки стала опечатка.
Обычно, когда вы пытаетесь открыть несуществующий файл, операционная система создает новый файл для вывода, но, к счастью, она не делает этого, когда вы обращаетесь к несуществующему файлу для ввода.
ofstream ofs("no-such-file"); // создает новый файл no-such-file
ifstream ifs("no-file-of-this-name"); // ошибка: поток ifs не нахо-
// дится в состоянии good()
11.3.2. Бинарные файлы
В памяти мы можем представить значение 123 как целое или как строку. Рассмотрим пример.
int n = 123;
string s = "123";
В первом случае число 123
интерпретируется как (двоичное) число. Объем памяти, который оно занимает, совпадает с объемом памяти, который занимает любое другое целое число (4
байта, т.е. 32
бита на персональном компьютере). Если вместо числа 123
мы выберем число 12345
, то оно по-прежнему будет занимать те же самые четыре байта. Во втором варианте значение 123
хранится как строка из трех символов. Если мы выберем строку "12345"
, то для ее хранения нам потребуются пять символов (плюс накладные расходы памяти на управление объектом класса string
). Проиллюстрируем сказанное, используя обычные десятичное и символьное представления, а не двоичное, как в памяти компьютера.
Когда мы используем символьное представление, то какой-то символ должен служить признаком конца числа, так же как на бумаге, когда мы записываем одно число 123456 и два числа 123 456. На бумаге для разделения чисел мы используем пробел. То же самое можно сделать в памяти компьютера.