Используйте тип time_t и tm struct из , а также фасеты даты и времени, предусмотренные в , для записи и чтения дат и времен (фасеты вскоре будут рассмотрены при обсуждении примера). См. пример 13.4.

Пример 13.4. Запись и чтение дат

#include

#include

#include

#include

#include

using namespace std;

void translateDate(istream& in, ostream& out) {

 // Создать считывающий дату объект

 const time get& dateReader =

  use_facet >(in.getloc());

 // Создать объект состояния который будет использован фасетом для

 // уведомления нас о возникновении проблемы

 ios_base::iostate state = 0;

 // Маркер конца

 istreambuf_iterator end;

 tm t; // Структура для представления времени (из )

 // Теперь, когда все подготовлено, считать дату из входного потока

 // и поместить ее в структуру времени.

 dateReader.get_date(in, end, in, state, &t);

 // В данный момент дата находится в структуре tm. Вывести ее в поток,

 // используя соответствующую локализацию. Убедитесь, что выводятся только

 // достоверные данные из t.

 if (state == 0 || state == ios_base::eofbit) { // Чтение выполнено успешно.

  const Time_put& dateWriter =

   use_facet >(out.getloc());

  char fmt[] = "%x";

  if (dateWriter.put{out, out, out.fill(),

   &t, &fmt[0], &fmt[2]).failed())

   cerr << "Unable to write to output stream.\n";

 } else {

  cerr << "Unable to read cin!\n";

 }

}

int main() {

 cin.imbue(locale("english"));

 cout.imbue(locale("german"));

 translateDate(cin, cout);

}

Эта программа выдает следующий результат

3/28/2005

28.03.2005

Обсуждение

Для правильной записи и чтения значений даты и времени необходимо знать некоторые детали проекта класса locale. Прочтите введение в эту главу, если вы еще не знакомы с концепциями локализаций и фасетов.

В C++ нет стандартного класса для представления даты и времени, а наиболее подходящими для этого типами являются time_t и структура tm из . Если требуется записывать и считывать даты с использованием средств стандартной библиотеки, вам придется любое нестандартное представление даты преобразовывать в структуру tm. Это имеет смысл, поскольку используемые вами реализации, вероятно, уже имеют встроенную поддержку форматирования дат с учетом местных особенностей.

Ранее я говорил, что фасет определяет некоторый аспект локализации, отражающий ее особенности. Более конкретно, фасет — это константная инстанциация шаблона класса символьного типа, поведение которого зависит от класса локализации, используемого при конструировании. В примере 13.4 я следующим образом создаю экземпляр фасета time_get.

const time_get& dateReader =

 use_facet >(in.getloc());

Шаблон функции use_facet находит заданный фасет для заданной локализации. Все стандартные фасеты являются шаблонами классов, которые принимают параметр символьного типа, и, поскольку мною считываются и записываются символы типа char, я инстанцирую мой класс time_get для char. Стандарт требует, чтобы реализация обеспечивала специализацию шаблона для char и wchar_t, поэтому они гарантированно существуют (хотя не гарантируется поддержка заданной локализации, кроме локализации С). Созданный мною объект time_get имеет спецификатор const, потому что предусмотренная реализацией функциональность локализации это набор правил форматирования различного вида данных в разных локализациях, и эти правила не могут редактироваться пользователем, поэтому состояние заданного фасета не должно изменяться в программном коде, где он используется.

Перейти на страницу:

Похожие книги