Строковый класс обеспечивает прямой доступ к символам строки (операция индексирования [ ] и метод at ()), а также последовательный доступ — с помощью прямых (методы begi п () и end ()) и обратных (методы гbegi п () и rend ()) итераторов.
Можно видеть большое разнообразие перегрузок одной и той же функции — это связано с тем, что требуется выполнять операции и со строками, и с символьными массивами (строками в стиле С). Поэтому практически каждый метод реализован как минимум в двух вариантах: с параметрами типов basic_string и charT*.
Создавать объекты-строки можно разнообразнейшими способами:
string s; // создает пустую строку
string s(cstr); // создает строку из строки в стиле С
string s(cstr, len); // создает строку из len символов строки в стиле С
string s(num, ch): // создает строку из num символов ch
string s(str); // создает строку-копию из строки str
string s(str, ind); // создает строку из символов строки str,
// начиная с индекса ind string s(str, ind, len); // создает строку длиной не более len символов
// строки str, начиная с индекса ind
string s(beg, end); // создает строку из интервала [beg, end)
// символов другой строки
Как видим, строка не может инициализироваться единичной символьной константой, например ' а'. Но, естественно, один из конструкторов является конструктором преобразования const charT* -> string. Обратное преобразование можно выполнять только явным образом с помощью трех методов:
• data () — возвращает содержимое строки в виде массива символов (в конце массива отсутствует завершающий символ ' \0');
• c_str() — возвращает содержимое строки в виде строки в стиле С (с завершающим символом ' \0');
• сору () — копирует содержимое строки в символьный массив-параметр метода (завершающий символ ' \ 0' отсутствует);
Функции d a t а () и c_s t г () возвращают указатель на внутренний буфер строки, поэтому вызывающая сторона не должна изменять символы или освобождать память.
Операция сцепления operator+ реализована как внешняя дружественная функция со всеми возможными сочетаниями типов параметров:
// Сцепление
template <class charT, class traits, class Allocator> basic_string operator+ (const basic_string&, con^t basic_string&);
template <class charT, class traits, class Allocator bas1c_str1ng operator* (const charT*, const basic_string&);
template <class charT, class traits, class Allocator>
basic_string operator* (charT, const basic_string&); template <class charT, class traits, class Allocator>
basic_string operator* (const basic_string&, const charT*); template <class charT, class traits, class Allocator>
basic_string operator* (const basic_string&, charT);
Кроме того, реализован полный набор операций сравнения; каждая операция реализована в трех вариантах со всеми возможными сочетаниями параметров. Например, сравнение на равенство обеспечивается следующими тремя функциями:
template <class charT, class traits, class Allocator> bool operator== (const basic_string&, const basic_string&);
template <class charT, class traits, class Allocator bool operator== (const charT*, const basic_string&);
template <class charT, class traits, class Allocator> bool operator== (const basic_string&, const charT*);
Как обычно, реализована функция обмена swap():
template <class charT, class traits, class Allocator> void swap(basic_string<charT,traits,Allocator>& a, basic_string<charT,traits,Allocator>& b);
И наконец, в виде внешних дружественных функций реализованы операции ввода-вывода строк, которые мы рассматривали в главе 14:
template<class charT, class traits, class Allocator> basic_istream<charT, traits>& operator>>(istream&, basic_string&);
template <class charT, class traits, class Allocator> basic_ostream<charT, traits>& operator<<(ostream&, const basic_string&);
template <class Stream, class charT, class traits, class Allocator> basic_istream<charT, traits>& getline(Stream&, basic_string&, charT);
|