C++. Сборник рецептов
Шрифт:
#include <cstdlib>
#include <string>
#include <cctype>
#include <functional>
using namespace std;
void textWrap(istream& in, ostream& out, size_t width) {
string tmp;
char cur = '\0';
char last = '\0';
size_t i = 0;
while (in.get(cur)) {
if (++i == width) {
ltrimws(tmp); // ltrim
как в рецепте
out << '\n' << tmp; // 4.1
i = tmp.length;
tmp.clear;
} else if (isspace(cur) && // Это конец
!isspace(last)) { // слова
out << tmp;
tmp.clear;
}
tmp += cur;
last = cur;
}
}
int main(int argc, char** argv) {
if (argc < 3)
return(EXIT_FAILURE);
int w = 72;
ifstream in(argv[1]);
ofstream out(argv[2]);
if (!in || !out)
return(EXIT_FAILURE);
if (argc == 4) w = atoi(argv[3]);
textWrap(in, out, w);
out.close;
if (out)
return(EXIT_SUCCESS);
else
return(EXIT_FAILURE);
}
Обсуждение
textWrap
читает по одному символы из входного потока. Каждый символ добавляется к временной строке tmp
до тех пор, пока не будет достигнут конец слов или максимальная длина строки. Если достигнут конец слова, а максимальная длина строки еще не достигнута, то временная строка записывается в выходной поток. В противном случае, если максимальная длина строки была превышена, в выходной поток записывается новая строка, пробел в начале временной строки удаляется, и строка записывается в выходной поток. Таким образом, textWrap
записывает в выходной поток столько, сколько можно, но не превышая максимальной длины строки. Вместо разделения слов она переносит все слово на новую строку. Пример 4.25 использует потоки почти так же, как и рецепт 4.15. За дополнительной информацией о потоках и их использовании обратитесь к этому рецепту.
Смотри также
Рецепт 4.15.
4.17. Подсчет числа символов, слов и строк в текстовом файле
Проблема
Требуется подсчитать число символов, слов и строк — или каких-либо других элементов текста — в текстовом файле.
Решение
Для
countStuff
, которая именно это и делает. Пример 4.26. Подсчет статистики по текстовому файлу
#include <iostream>
#include <fstream>
#include <cstdlib>
#include <cctype>
using namespace std;
void countStuff(istream& in,
int& chars, int& words, int& lines) {
char cur = '\0';
char last = '\0';
chars = words = lines = 0;
while (in.get(cur)) {
if (cur == '\n' ||
(cur == '\f' && last == '\r'))
lines++;
else chars++;
if (!std::isalnum(cur) && // Это конец
std::isalnum(last)) // слова
words++;
last = cur;
}
if (chars > 0) { // Изменить значения слов
if (std::isalnum(last)) // и строк для специального
words++; // случая
lines++;
}
}
int main(int argc, char** argv) {
if (argc < 2)
return(EXIT _FAILURE);
ifstream in(argv[1]);
if (!in)
exit(EXIT_FAILURE);
int c, w, l;
countStuff(in, c, w, l);
cout << "символов: " << c << '\n';
cout << "слов: " << w << '\n';
cout << "строк: " << l << '\n';
}
Обсуждение
Этот алгоритм очень прост. С символами все просто: увеличивайте счетчик символов при каждом вызове
get
для входного потока. Со строками все не намного сложнее, так как способ представления концов строк зависит от операционной системы. К счастью, обычно это либо символ новой строки (\n
), либо последовательность из символов возврата каретки и перевода строки (\r\n
). Отслеживая текущий и предыдущий символы, можно легко обнаружить вхождения этой последовательности. Со словами все проще или сложнее, в зависимости от определения того, что такое «слово».
Поделиться:
Популярные книги
Неучтенный. Дилогия
Неучтенный
Фантастика:
боевая фантастика
попаданцы
7.98
рейтинг книги
Попаданка
Любовные романы:
любовно-фантастические романы
5.00
рейтинг книги
Я еще не барон
1. Дорогой барон!
Фантастика:
боевая фантастика
попаданцы
аниме
5.00
рейтинг книги
Жена проклятого некроманта
Фантастика:
фэнтези
6.60
рейтинг книги
Камень. Книга шестая
6. Камень
Фантастика:
боевая фантастика
7.64
рейтинг книги
Барон нарушает правила
3. Закон сильного
Фантастика:
фэнтези
попаданцы
аниме
5.00
рейтинг книги
Комбинация
2. Сын Петра
Фантастика:
попаданцы
альтернативная история
5.00
рейтинг книги
Испытание Огня
3. Академия Стихий
Фантастика:
фэнтези
9.43
рейтинг книги
Идеальный мир для Лекаря 25
25. Лекарь
Фантастика:
фэнтези
юмористическое фэнтези
аниме
5.00
рейтинг книги
Ведьмак. Перекресток воронов
Фантастика:
фэнтези
5.00
рейтинг книги
Сердце Дракона. Том 20. Часть 1
20. Сердце дракона
Фантастика:
фэнтези
боевая фантастика
городское фэнтези
5.00
рейтинг книги
Вечная Война. Книга II
2. Вечная война.
Фантастика:
юмористическая фантастика
космическая фантастика
8.37
рейтинг книги
Клан
2. Долгий путь домой
Фантастика:
боевая фантастика
космическая фантастика
5.60
рейтинг книги
Герцог и я
1. Бриджертоны
Любовные романы:
исторические любовные романы
8.92