C++. Сборник рецептов - Д. Стефенс
Шрифт:
Интервал:
Закладка:
#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();
}
Этот пример принимает три аргумента: входной файл, выходной файл и ширину выровненного по правому краю текста. Входной файл может иметь следующий вид.
With automatic download of Microsoft's (Nasdaq:
MSFT) enormous SP2 security patch to the Windows
XP operating system set to begin the industry
still waits to understand its ramifications. Home
users that have their preferences set to receive
operating system updates as they are made
available by Microsoft may be surprised to learn
that some of the software they already run on
their systems could be disabled by SP2 or may run
very differently.
Вывод будет иметь следующий вид.
With automatic download of Microsoft's (Nasdaq:
MSFT) enormous SP2 security patch to the Windows
XP operating system set to begin the industry
still waits to understand its ramifications. Home
users that have their preferences set to receive
operating system updates as they are made
available by Microsoft may be surprised to learn
that some of the software they already run on
their systems could be disabled by SP2 or may run
very differently.
Второй пример текста выровнен по правому краю и имеет в ширину 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.
Смотри такжеРецепты 4.15 и 4.16.
4.22. Автозамена текста при изменении буфера
ПроблемаИмеется класс, который представляет некий тип текстового поля или документа, и по мере добавления в него текста требуется автоматически корректировать неправильно написанные слова, как это делает функция Autocorrect (Автозамена) в Microsoft Word.
РешениеЭто можно реализовать в относительно небольшом коде, если использовать map, который определен в <map>, string и различные возможности стандартной библиотеки. Пример 4.31 показывает, как это делается.
Пример 4.31. Автозамена текста
#include <iostream>
#include <string>
#include <cctype>
#include <map>
using namespace std;
typedef map<string, string> StrStrMap;
// Класс для хранения текстовых полей
class TextAutoField {
public:
TextAutoField(StrStrMap* const p) : pDict_(p) {}
~TextAutoField() {}
void append(char c);
void getText(string& s) {s = buf_;}
private:
TextAutoField();
string buf_;
StrStrMap* const pDict ;
};
// Добавление с автозаменой
void TextAutoField::append(char c) {
if ((isspace(c) || ispunct(c)) && // Выполнять автоза-
buf_.length() > 0 && // мену, только когда вводятся
!isspace(buf_[buf_.length() - 1])) { // ws или punct
string::size_type i = buf_.find_last_of(" fnrtv");
i = (i == string::npos) ? 0 : ++i;
string tmp = buf_.substr(i, buf_.length() - i);
StrStrMap::const_iterator p = DDict_->find(tmp);
if (p != pDict_->end()) { // Нашли, так что стираем
buf_.erase(i, buf_.length() - i); // и заменяем
buf_ += p->second;
}
}
buf_ += с;
}
int main() {
// Создаем map
StrStrMap dict;
TextAutoField txt(&dict);
dict["taht"] = "that";
dict["right"] = "wrong";
dict["bug"] = "feature";
string tmp = "He's right, taht's a bug.";
cout << "Оригинальная версия: " << tmp << 'n';
for (string::iterator p = tmp.begin(); p != tmp.end(); ++p) {