Этот пример делает несколько предположений о формате входного текста, так что внимательно прочтите следующий раздел.
Обсуждение
addMargins
предполагает, что ввод выглядит примерно так.
The data is still inconclusive. But the weakness
in job creation and the apparent weakness in
high-paying jobs may be opposite sides of a coin.
Companies still seem cautious, relying on
temporary workers and anxious about rising health
care costs associated with full-time workers
Этот
текст содержит переносы в позиции 50 символов (см. рецепт 4.16) и выровнен по левому краю (см. рецепт 4.20).
addMargins
также предполагает, что требуется, чтобы вывод выглядел подобно следующему, который использует для обозначения полей вместо пробелов точки.
.......The data is still inconclusive. But the weakness..............
.......in job creation and the apparent weakness in..................
.......high-paying jobs may be opposite sides of a coin..............
.......Companies still seem cautious, relying on.....................
.......temporary workers and anxious about rising health.............
.......care costs associated with full-time workers..................
По умолчанию левое поле содержит восемь символов, а общая длина строки составляет 72 символа. Конечно, если известно, что входной текст будет всегда выровнен по левому или правому краю, то можно просто дополнить оба конца каждой строки таким количеством символов, которое требуется. В любом случае логика очень проста. Многие методики, используемые в этом рецепте, уже описывались (потоки, дополнение
string
), так что я не буду здесь на них останавливаться. Единственная новая функция здесь — это
getline
.
Если требуется прочитать сразу целую строку текста или, более точно, прочитать текст до определенного разделителя, используйте шаблон функции
getline
, определенный в
<string>
, как это сделано в примере 4.28.
getline(in, tmp, '\n');
getline
читает символы из входного потока и добавляет их в
tmp
до тех пор, пока не встретится разделитель
'\n'
, который в
tmp
не добавляется.
basic_istream
содержит метод с таким же именем, но с другим поведением. Он сохраняет свой вывод в символьном буфере, а не в
string
. В данном случае я решил использовать преимущества метода из
string
, так как мне не хотелось читать строку в символьный буфер, а затем копировать ее в
string
. Таким образом, я использовал
getline
в версии
string
.
Смотри также
Рецепты 4.16 и 4.20.
4.20. Выравнивание текста в текстовом файле
Проблема
Требуется выровнять текст по правому или левому краю.
Решение
Используйте потоки и стандартные флаги форматирования потоков
right
и
left
, являющиеся частью
ios_base
, определенного в
<ios>
. Пример 4.29 показывает, как они работают.
Пример 4.29. Выравнивание текста
#include <iostream>
#include <fstream>
#include <string>
#include <cstdlib>
using namespace std;
int main(int argc, char** argv) {
if (argc < 3)
return(EXIT_FAILURE);
ifstream in(argv[1]);
ofstream out(argv[2]);
int w = 72;
if (argc == 4)
w = atoi(argv[3]);
string tmp;
out.setf(ios_base::right); // Указать потоку на
// выравнивание по правому краю
while (!in.eof) {
out.width(w); // Сбросить ширину после
getline(in, tmp, "\n"); // каждой записи
out << tmp << '\n';
}
out.close;
}
Этот пример принимает три аргумента: входной файл, выходной файл и ширину выровненного по правому краю текста. Входной файл может иметь следующий вид.