Второй пример текста выровнен по правому краю и имеет в ширину 50 символов.
Обсуждение
Шаблон класса
ios_base
содержит большое количество флагов форматирования числовых и текстовых данных, читаемых из потоков или записываемых в них. Два флага, управляющих выравниванием текста, — это
right
и
left
. Они являются
static const
– членами
ios_base
и имеют тип
fmtflags
(который зависит от реализации).
Все это хозяйство определено в
<ios>
.
Чтобы установить флаги форматирования, используйте
ios_base::setf
. Она объединяет переданные в нее и уже установленные ранее флаги потока с помощью операции OR (ИЛИ). Например, эта строка включает выравнивание по правому краю:
out.setf(std::ios_base::right);
Но выравнивание по правому краю не имеет смысла без установки правого поля, по которому требуется выравнивать. Чтобы установить это поле, используйте
ios_base::width
, как здесь.
out.width(w);
Этот код устанавливает ширину выходного поля для передаваемого значения, что означает, что при выравнивании текста по правому краю начало строки будет дополняться пробелами так, чтобы ее правый край достиг правого поля. Заметьте, что ширину я задаю в цикле, в то время как флаг
right
я выставляю перед ним. Это требуется делать потому, что после каждой записи в поток ширина сбрасывается в ноль. Флаги форматирования после записи не сбрасываются, так что их можно указать только один раз.
Однако всегда следует быть аккуратным и точным, так что при использовании флагов форматирования требуется сделать еще одну вещь: убрать их за собой.
Часто потоки, в которые производится запись, не принадлежат записывающему коду, особенно при написании библиотеки или API общего назначения. Например, если написать функцию журналирования, которая принимает выходной поток и
string
, изменяет
string
, устанавливает флаги форматирования и записывает
string
в поток, то могут возникнуть нежелательные побочные эффекты. После того как клиентский код вызывает эту функцию журналирования, его поток содержит другой набор флагов форматирования. Решением является копирование старых флагов и восстановление их по окончании работы.
Например, функция журналирования ошибок может выглядеть вот так.
using namespace std;
void logFrror(ostream& out, const string& s) {
string tmp(s);
tmp.insert(0, "ERROR: ");
ios_base::fmtflags figs = // setf возвращает
out.setf(ios_base::left); // флаги, которые уже
// были установлены
out.width(72);
out << tmp << '\n';
out.flags(flgs); // вернуть оригинальные
}
Метод
flags
работает аналогично
setf
, но не объединяет с помощью OR переданные ему флаги с уже установленными, а заменяет их. Таким образом, при вызове
flags
и передаче ему оригинальных флагов форматирования можно быть уверенным, что флаги будут восстановлены.
4.21. Замена в текстовом файле последовательностей пробелов на один пробел
Проблема
Имеется текстовый файл с последовательностями пробелов различной длины и требуется заменить каждое вхождение такой последовательности на единственный пробел.
Решение
Для чтения непрерывной последовательности непробельных символов из потока в строку используйте шаблон функции
operator>>
, определенный в
<string>
. Затем используйте его двойника
operator<<
, который записывает каждую из этих последовательностей в выходной поток, и после каждой из них добавьте по одному пробелу. Пример 4.30 дает краткий пример этой методики.
Пример 4 30. Замена последовательностей пробелов на один пробел
#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main(int argc, char** argv) {
if (argc < 3)
return(EXIT_FAILURE);
ifstream in(argv[1]);
ofstream out(argv[2]);
if (!in || !out)
return(EXIT_FAILURE);
string tmp;
in >> tmp; // Прочитать первое слове
out << tmp; // Записать его в выходной поток
while (in >> tmp) { // operator>> игнорирует пробелы, так что все, что
out << ' '; // я должен сделать, - это записать пробел и каждую
out << tmp; // последовательность «непробелов»
}
out.close;
}
Обсуждение
Это просто сделать, если использовать потоки и строки. Даже если требуется реализовать другой вариант этого — например, чтобы сохранить переходы на новую строку, — эта методика будет работать. Если требуется добавить переходы на новые строки, для их расстановки в нужных местах используйте решение, представленное в рецепте 4.16.