13.4. Запись и чтение денежных значений
Требуется записать в поток или прочитать из него денежное значение.
Используйте фасеты money_put
и money_get
для записи или чтения денежных значений, как показано в примере 13.6.
#include <iostream>
#include <locale>
#include <string>
#include <sstream>
using namespace std;
long double readMoney(istream& in, bool intl = false) {
long double val;
// Создать фасет для чтения
const money_get<char>& moneyReader =
use_facet<money_get<char> >(in.getloc());
// Маркер конца
istreambuf iterator<char> end;
// Переменная состояния для обнаружения ошибок
ios_base::iostate state = 0;
moneyReader.get(in, end, intl, in, state, val);
// если что-то не получилось, будет установлен бит неудачного завершения
if (state != 0 && !(state & ios_base::eofbit))
throw 'Couldn't read money!
';
return(val);
}
void writeMoney(ostream& out, long double val, bool intl = false) {
// Создать фасет для записи
const money_put<char>& moneyWriter =
use_facet<money_put<char> >(out.getloc());
// Записать данные в поток. Вызвать failed() (возвращает итератор
// ostreambuf_iterator), чтобы можно было обнаружить ошибку.
if (moneyWriter.put(out, intl, out, out.fill(), val).failed())
throw 'Couldn't write money!
';
}
int main() {
long double val = 0;
float exchangeRate = 0.775434f; // Курс доллара по отношению к евро
locale locEn('english');
locale locFr('french');
cout << 'Dollars: ';
cin.imbue(locEn);
val = readMoney(cin, false);
cout.imbue(locFr);
// Установить флаг showbase, чтобы выводить символ валюты
cout.setf(ios_base::showbase);
cout << 'Euros: ';
writeMoney(cout, val = exchangeRate, true);
}
Если выполнить программу примера 13.6, можно получить следующий результат.
Dollars: $100
Euros: EUR77,54
Фасеты money_put
и money_get
записывают форматированные денежные значения в поток и считывают их из потока. Они работают почти так же, как фасеты даты/времени и числовые фасеты, описанные в предыдущих рецептах. Стандарт требует, чтобы были их реализации для стандартных символов и расширенного набора символов, например money_put<char>
и money_put<wchar_t>
. Как и для других фасетов, функции записи и чтения многословны, но, применив их несколько раз, легко запоминаешь параметры. money_get
и money_put
используют класс moneypunct
, содержащий информацию о форматировании.
Сначала рассмотрим запись денежных значений в поток. Отображение денежной суммы состоит из нескольких частей: знака валюты, знака плюс или минус, разделителя тысяч и десятичной точки. Все они, кроме десятичной точки, могут отсутствовать.
Вы создаете объект money_put
с типом символа и локализацией следующим образом.
const money_put<char>& moneyWriter =
use_facet<money_put<char> >(out.getloc());
Стандарт требует наличия версий как для char
, так и для wchar_t
. Разумно использовать локализацию потока, в который осуществляется запись, чтобы избежать несогласованности, возникающей при попытке синхронизации потока и объекта money_put
. На следующем шаге вызовите метод put для записи денежного значения в поток вывода.
if (moneyWriter.put(out, // Итератор вывода
intl, // bool: использовать формат intl?
out, // ostream&
out.fill(), // использовать символ заполнителя
val) // денежное значение, тип long double
.failed()) throw 'Couldn't write money!
';
Функция money_put::put
записывает денежное значение в переданный ей поток вывода, используя локализацию, с которой был создан объект money_put
. money_put::put
возвращает итератор ostreambuf_iterator
, который ссылается на позицию за последним выведенным символом; этот итератор имеет функцию-член failed
, позволяющую зафиксировать ситуацию, когда итератор оказывается испорченным.
Все параметры money_put::put
не требуют дополнительных пояснений, кроме, возможно, второго (аргумент intl
в примере). Он имеет тип bool
и показывает, будет использоваться символ валюты (например, $, €) или трехбуквенное международное обозначение валюты (например, USD, EUR). Для использования символа валюты установите его в значение false
, а для использования международного обозначения валюты — в значение true
.
При записи денежных значений в поток вывода можно задавать некоторые параметры потока, которые управляют форматированием. Ниже описывается каждый параметр и объясняется его воздействие на вывод денежного значения.
ios_base::internal