string s = "One fish, two fish, red fish, blue fish";
string p = "fish";
removeSubstrs(s, p);
cout << s << '\n';
}
Здесь
всю важную работу выполняет метод
erase basic_string
. В
<string>
он перегружен три раза. Использованная в примере 4.19 версия принимает индекс, с которого требуется начать удаление, и число удаляемых символов. Другая версия принимает в качестве аргументов начальный и конечный итераторы, а также есть версия, которая принимает единственный итератор и удаляет элемент, на который он указывает. Чтобы обеспечить оптимальную производительность при планировании удаления нескольких последовательных элементов, используйте первые две версии и не вызывайте
s.erase(iter)
несколько раз для удаления каждого из идущих подряд элементов. Другими словами, используйте методы, работающие с диапазонами, а не с одним элементом, особенно в случае тех методов, которые изменяют содержимое строки (или последовательности). В этом случае вы избежите дополнительных вызовов функции
erase
для каждого элемента последовательности и позволите реализации
string
более грамотно управлять ее содержимым.
4.12. Преобразование строки к нижнему или верхнему регистру
Проблема
Имеется строка, которую требуется преобразовать к нижнему или верхнему регистру.
Решение
Для преобразования символов к нижнему или верхнему регистру используйте функции
toupper
и
tolower
из заголовочного файла
<cctype>
. Пример 4.20 показывает, как использовать эти функции. Смотри также обсуждение альтернативных методик.
Пример 4.20. Преобразование регистра строки
#include <iostream>
#include <string>
#include <cctype>
#include <cwctype>
#include <stdexcept>
using namespace std;
void toUpper(basic_string<char>& s) {
for (basic_string<char>::iterator p = s.begin;
p != s.end; ++p) {
*p = toupper(*p); // toupper is for char
}
}
void toUpper<basic_string<wchar_t>& s) {
for (basic_string<wchar_t>::iterator p = s.begin;
p != s.end; ++p) {
*p = towupper(*p); // towupper is for wchar_t
}
}
void toLower(basic_string<char>& s) {
for (basic_string<char>::iterator p = s.begin;
p != s.end; ++p) {
*p = tolower(*p);
}
}
void toLower(basic_string<wchar_t>& s) {
for (basic_string<wchar_t>::iterator p = s.begin;
p != s.end; ++p) {
*p = towlower(*p);
}
int main {
string s = "shazam";
wstring ws = L"wham";
toUpper(s); toUpper(ws);
cout << "s = " << s << endl;
wcout << "ws = " << ws << endl;
toLower(s);
toLower(ws);
cout << "s = " << s << endl;
wcout << "ws = " << ws << endl;
}
Этот
код производит следующий вывод.
s = SHAZAM
ws = WHAM
s = shazam
ws = wham
Обсуждение
Кто-то может подумать, что стандартный класс
string
содержит метод, преобразующий всю строку к верхнему или нижнему регистру, но на самом деле это не так. Если требуется преобразовать строку символов к верхнему или нижнему регистру, это требуется делать самостоятельно.
Неудивительно, что имеется несколько способов преобразования регистра строки (и когда я говорю «строки», то имею в виду последовательность символов как узких, так и широких). Простейшим способом сделать это является использование одной из четырех функций преобразования символов
toupper
,
towupper
,
tolower
и
towlower
. Первая форма этих функций работает с узкими символами, а вторая форма (с дополнительной буквой
w
) является ее эквивалентом для широких символов.
Каждая из этих функций преобразует регистр символа, используя текущие правила локали для преобразования регистра. Верхний и нижний регистры зависят от символов, используемых в текущей локали. Некоторые символы не имеют верхнего или нижнего регистра, и в этом случае указанные функции возвращают переданный им символ. За дополнительной информацией о локалях обратитесь к главе 13. Возможности C++ по работе с различными локалями довольно сложны, и я не могут уделить им сейчас достаточно места.
Выполнение собственно преобразования символов просто. Рассмотрим функцию
toUpper
из примера 4.20.
void toUpper(basic_string<char>& s) {
for (basic_string<char>::iterator p = s.begin;
p != s.end; ++p) {
*p = toupper(*p);
}
}
Строка, выделенная жирным, выполняет всю работу. Версия для широких символов почти идентична.