Для ввода-вывода символов и строк на кириллице необходимо установить кириллицу в качестве локального контекста, или локаля (locale ). Локальный контекст (см. п. п. 22.1.1 в [1]) — это объект, инкапсулирующий национальные особенности внешнего представления данных вроде символа-разделителя целой и дробной частей в числах, формата даты и тому подобных «мелочей» (см. п. п. 22.1.1.1.1 в [1]). Заодно в локальном контексте задается и нужная кодировка. Вообще-то объекты-контексты устроены довольно сложно — в стандарте им посвящено почти 50 страниц (см. п. 22 в [1]), но мы рассмотрим только простейшие применения для работы с потоками.
Для работы с контекстами нужно прописать в программе оператор #include <locale>
После этого можно объявлять в программе объекты-контексты, например:
locale american; // установки по умолчанию
Установки по умолчанию ориентированы, естественно, на США. Однако ни объявление контекстов, ни стандартные американские установки нас не интересуют. Локальный контекст устанавливается для потока. Чтобы выводить символы кириллицы на консоль, мы должны установить русский локальный контекст для потока wcout, а чтобы вводить символ кириллицы, — русский контекст для потока wci п. Это делается с помощью метода imbue() потока (см. п. п. 27.4.2.3 в [1]), имеющего следующий прототип:
locale imbue(const locale& L);
Параметром служит объект-контекст, но его не обязательно объявлять в программе — можно использовать временный анонимный объект, например:
wcout.imbue(locale("rus_rus.866"));
В данном случае русский контекст установлен для стандартного широкого потока wcout. Аргумент контекста — строка "rus_rus.866". Эта строка называется именем локального контекста, и ее вид зависит от реализации . Это имя можно получить в виде строки типа string с помощью метода name() класса locale:
locale loc ("rus_rus.866"); cout << loc.name( ) << endl;
В результате на экран будет выведено следующее: Russi an_Russian.866
Это — полное имя контекста, мы же писали сокращенное. Первое слово «Russian* означает русский язык, второе — страну, число 866 — кодовую страницу.
После установки русского контекста на экран в консольное окно нормально выводятся русскоязычные сообщения, например:
wstring s(L"Привет!");
wcout << 1_"Снова привет!" << endl;
wcout << s << endl;
Однако правильно выводятся только широкие русскоязычные строковые константы — в отличие от англоязычных строковых констант, узкие русскоязычные строковые константы выводятся неправильно.
ПРИМЕЧАНИЕ
Можно установить русский контекст и для узких потоков. Однако в системе Windows консольные программы работают в специальном консольном окне, для которого система по умолчанию устанавливает кодовую страницу 866 как для ввода, так и для вывода. Когда же мы набираем текст программы в окне интегрированной среды, то устанрвлена кодовая страница 1251 и русскоязычные константы записываются в программу в этой кодировке. Поэтому даже при установлении русского контекста для потока cout русскоязычные текстовые константы выводятся неверно.
Для ввода кириллицы нужно точно таким же способом задать российский контекст для входного широкого потока wcin:
wcin.imbue(locale("rus_rus.866"));
После этого можно будет вводить русскоязычные строки и символы обычными способами, которые мы уже рассматривали, например:
wstring s;
getline(wcin, s); // ввод широкой строки с пробелами
wchar_t g[100]:
wcin.getline(g,99); // ввод символьного массива с пробелами
wcin.get(ch); // ввод широкого символа
ch = wcin.getO; // ввод широкого символа
Все введенные символы нормально выводятся на экран.
Текущий локальный контекст можно сохранить как объект-локаль с помощью метода getlocQ, который имеет прототип
locale getlocO const:
Можно установить «классический» локальный контекст С с помощью метода static const locale& classic( );
Если вам интересно узнать http://mob-mobile.ru/statya/1569-chem-otlichaetsya-ayfon-ot-aypoda.html тогда зайдите сюда.
Можно установить некоторый контекст по умолчанию методом static locale global(const locale& L); Применять эти методы можно так:
locale current = wcout.getlocO; // сохранение текущего контекста
wcout.imbue(locale::classic()); // установка классического контекста
wcout.imbue(locale("CM)); // тоже установка классического контекста
wcout.imbue(current); // восстановление прежнего контекста
locale::global(current); // установка контекста по умолчанию
|