можно внести пару усовершенствований. Во-первых, для простоты
StringTokenizer
написан так, что он работает только с простыми строками — другими словами, строками из узких символов. Если требуется, чтобы один и тот же класс работал как с узкими, так и с широкими символами, параметризуйте тип символов, как это сделано в предыдущих рецептах. Другим улучшением является расширение
StringTokenizer
так, чтобы он обеспечивал более дружественное взаимодействие с последовательностями и был более гибок. Вы всегда можете сделать это сами, а можете использовать имеющийся класс разбиения на лексемы. Проект Boost содержит
класс
tokenizer
, делающий все это. За подробностями обратитесь к www.boost.org.
Смотри также
Рецепт 4.24.
4.8. Объединение нескольких строк
Проблема
Имея последовательность строк, такую как вывод примера 4.10, вам требуется объединить их в одну длинную строку, возможно, с разделителями.
Решение
В цикле переберите всю последовательность строк и добавьте каждую из них в выходную строку. В качестве входа можно обрабатывать любую стандартную последовательность. Пример 4.13 использует
Пример 4.13 содержит одну методику, которая несколько отличается от предыдущие примеров. Посмотрите на эту строку.
for (vector<string>::const_iterator p = v.begin;
Предыдущие примеры работы со строками использовали
iterator
'ы без части «const», но здесь без этого не обойтись, так как
v
объявлен как ссылка на объект
const
. Если имеется объект контейнера
const
, то для доступа к его элементам можно использовать только
const_iterator
. Это так потому, что простой
iterator
позволяет записывать в объект, на который он указывает, что, конечно, нельзя делать в случае с объектами контейнера типа
const
.
v
объявлен как
const
по двум причинам. Во-первых,
я знаю, что я не собираюсь изменять его содержимое, так что я хочу, чтобы компилятор выдал сообщение об ошибке, если это произойдет. Компилятор гораздо лучше меня в деле поиска таких вещей, особенно когда к такому присвоению приводит тонкая семантическая или синтаксическая ошибка. Во-вторых, я хочу показать пользователям этой функции, что я ничего не делаю с их контейнером, и
const
— это великолепный способ сделать это. Теперь я просто должен создать обобщенную версию, которая работает с различными типами символов.
Как и в рецепте 4.6, превращение
join
в общий шаблон функции очень просто. Все, что требуется сделать, — это изменить заголовок, параметризовав тип символов, как здесь:
template<typename T>
void join(const std::vector<std::basic_string<T> >& v, T c,
std::basic_string<T>& s)
Но
vector
может оказаться не единственным возможным входом функции. Вам может потребоваться объединить строки в стиле С. Класс
string
C++ предпочтительнее строк в стиле С, так что если возникает такая задача, объединяйте их в C++
string
. После этого всегда можно получить версию С, вызвав метод
string c_str
, который возвращает указатель
const
на завершающийся нулем массив символов.
Пример 4.14 предлагает общую версию
join
, которая объединяет массив символов в
string
. Так как новая общая версия параметризована по типу символов, она будет работать как для массивов узких, так и для массивов широких символов.
Пример 4.14 Объединение строк в стиле C
#include <string>
#include <iostream>
const static int MAGIC_NUMBER = 4;
template<typename T>
void join(T* arr[], size_t n, T c, std::basic_string<T>& s) {
s.clear;
for (int i = 0; i < n; ++i) {
if (arr[i] != NULL)
s += arr[i];
if (i < n-1) s += c;
}
}
int main {
std::wstring ws;
wchar_t* arr[MAGIC_NUMBER];
arr[0] = L"you";
arr[1] = L"ate";
arr[2] = L"my";
arr[3] = L"breakfast";
join(arr, MAGIC_NUMBER, L'/', ws);
}
4.9. Поиск в строках
Проблема
Требуется выполнить поиск в строке. Это может быть поиск одного символа, другой строки или одного из (или одного не из) неупорядоченного набора символов. И по каким-либо причинам требуется выполнять поиск в определенном порядке, например первое или последнее вхождение или первое или последнее вхождения относительно какого- либо положения в строке.