C++. Сборник рецептов - [81]

Шрифт
Интервал

Решение

При работе с датами и временем часто возникает необходимость ограничить целые значения диапазоном допустимых значений (т.е для секунд в минуте — от 0 до 59, для часов в сутках от 0 до 23, для дней в году — от 0 до 365). Вместо того чтобы каждый раз проверять эти значения при их передаче в функцию, предпочтительной является их автоматическая проверка с помощью перегруженного оператора присвоения. Так как имеется очень большое количество таких типов, следует реализовать один тип, который сможет работать с подобной проверкой для различных числовых диапазонов. Пример 5.10 представляет реализацию шаблона класса >ConstrаinedValue, который облегчает задание диапазона целых чисел и определение других ограниченных типов.

Пример 5.10. constrained_value.hpp

>#ifndef CONSTRAINED_VALUE_HPP

>#define CONSTRAINED_VALUE_HPP


>#include

>#include


>using namespace std;


>template

>struct ConstrainedValue {

>public:

> // открытые typedef

> typedef typename Policy_T policy_type;

> typedef typename Policy_T::value_type value_type;

> typedef ConstrainedValue self;


> // конструктор по умолчанию

> ConstrainedValue() : m(Policy_T::default_value) {}

> ConstrainedValue(const self& x) : m(x.m) {}

> ConstrainedValue(const value_type& x) { Policy_T::assign(m, x); }

> operator value_type() const { return m; }


> // использует функцию присвоения, определенную политикой

> void assign(const value_type& x) {

>  Policy_T::assign(m, x);

> }


> // операции присвоения

> self& operator=(const value_type& x) { assign(x); return *this; }

> self& operator+=(const value_type& x) { assign(m + x) return *this; }

> self& operator-=(const value_type& x) { assign(m - x) return *this; }

> self& operator*=(const value_type& x) { assign(m * x); return *this; }

> self& operator/=(const value_type& x) { assign(m / x); return *this; }

> self& operator%=(const value_type& x) { assign(m % x); return *this; }

> self& operator>>=(int x) { assign(m >> x); return *this; }

> self& operator<<=(int x) { assign(m << x); return *this; }


> // унарные операторы

> self operator-() { return self(-m); }

> self operator+() { return self(-m); }

> self operator!() { return self(!m); }

> self operator~() { return self(~m); }


> // бинарные операторы

> friend self operator+(self x, const value_type& y) { return x += y; }

> friend self operator-(self x, const value_type& y) { return x -= y; }

> friend self operator*(self x, const value_type& y) { return x *= y; }

> friend self operator/{self x, const value_type& y) { return x /= y; }

> friend self operator%(self x, const value_type& y) { return x %= y; }

> friend self operator+(const value_type& y, self x) { return x += y; }

> friend self operator-(const value_type& y, self x) { return x -= y; }

> friend self operator*(const value_type& y, self x) { return x *= y; }

> friend self operator/(const value_type& y, self x) { return x /= y; }

> friend self operator%(const value_type& y, self x) { return x %= y; }

> friend self operator>>(self x, int y) { return x >>= y; }

> friend self operator<<(self x, int y) { return x <<= y; }


> // потоковые операторы

> friend ostream& operator<<(ostream& o, self x) { о << x.m; return o; }

> friend istream& operator>>(istream& i, self x) {

>  value_type tmp; i >> tmp; x.assign(tmp); return i;

> }


> // операторы сравнения

> friend bool operator<(const self& x, const self& y) { return x.m < y.m; }

> friend bool operator>(const self& x, const self& y) { return x.m > y.m; }

> friend bool operator<=(const self& x, const self& y) { return x.m <= y.m; }

> friend bool operator>=(const self& x, const self& y) { return x.m >= y.m; }

> friend bool operator==(const self& x, const self& y) { return x.m == y.m; }

> friend bool operator!=(const self& x, const self& y) { return x.m != y.m; }

>private:

> value_type m;

>};


>template

>struct RangedIntPolicy {

> typedef int value_type;

> const static value_type default_value = Min_N;


> static void assign(value_type& lvalue, const value_type& rvalue) {

>  if ((rvalue < Min_N) || (rvalue > Max_N) {

>   throw range_error("out of valid range");

>  }

>  lvalue = rvalue;

> }

>};

>#endif

Программа в примере 5.11 показывает, как использовать тип >ConstrainedValue.

Пример 5.11. Использование constrained_value.hpp

>#include "constrained_value.hpp"


>typedef ConstrainedValue< RangedIntPolicy<1582, 4000> > GregYear;

>typedef ConstrainedValue< RangedIntPolicy<1, 12> > GregMonth;

>typedef ConstrainedValue< RangedIntPolicy<1, 31> > GregDayOfMonth;


>using namespace std;


>void gregOutputDate(GregDayOfMonth d, GregMonth m, GregYear y) {

> cout << m << "/" << d << "/" << y << endl;

>}


>int main() {

> try {

>  gregOutputDate(14, 7, 2005);

> } catch(...) {

>  cerr << "Оп, не должны сюда попасть << endl;

> }

> try {

>  gregOutputDate(1, 5, 1148);

>  cerr << "Оп, не должны сюда попасть" << endl;

> } catch(...) {

>  cerr << "Уверены, что надо использовать григорианский календарь?" << endl;

> }

>}

Вывод программы из примера 5.11 имеет вид:

>7/14/2005

>Уверены, что надо использовать григорианский календарь?

Обсуждение

Ограниченные типы значений обычно используются при работе с датами и временем, так как многие значения, связанные с датами/временем, — это целые числа, которые должны находиться в определенных диапазонах (например, месяц должен быть в интервале [0,11], а день месяца должен быть в интервале [0,30]). Проверять вручную параметр каждой функции на допустимый диапазон очень долго и чревато ошибками. Просто представьте, что требуется внести глобальное изменение в то, как программа, содержащая миллион строк кода, обрабатывает ошибки диапазона дат!


Рекомендуем почитать
Изучаем Java EE 7

Java Enterprise Edition (Java EE) остается одной из ведущих технологий и платформ на основе Java. Данная книга представляет собой логичное пошаговое руководство, в котором подробно описаны многие спецификации и эталонные реализации Java EE 7. Работа с ними продемонстрирована на практических примерах. В этом фундаментальном издании также используется новейшая версия инструмента GlassFish, предназначенного для развертывания и администрирования примеров кода. Книга написана ведущим специалистом по обработке запросов на спецификацию Java EE, членом наблюдательного совета организации Java Community Process (JCP)


Pro Git

Разработчику часто требуется много сторонних инструментов, чтобы создавать и поддерживать проект. Система Git — один из таких инструментов и используется для контроля промежуточных версий вашего приложения, позволяя вам исправлять ошибки, откатывать к старой версии, разрабатывать проект в команде и сливать его потом. В книге вы узнаете об основах работы с Git: установка, ключевые команды, gitHub и многое другое.В книге рассматриваются следующие темы:основы Git;ветвление в Git;Git на сервере;распределённый Git;GitHub;инструменты Git;настройка Git;Git и другие системы контроля версий.


Java 7

Рассмотрено все необходимое для разработки, компиляции, отладки и запуска приложений Java. Изложены практические приемы использования как традиционных, так и новейших конструкций объектно-ориентированного языка Java, графической библиотеки классов Swing, расширенной библиотеки Java 2D, работа со звуком, печать, способы русификации программ. Приведено полное описание нововведений Java SE 7: двоичная запись чисел, строковые варианты разветвлений, "ромбовидный оператор", NIO2, новые средства многопоточности и др.


Фундаментальные алгоритмы и структуры данных в Delphi

Книга "Фундаментальные алгоритмы и структуры данных в Delphi" представляет собой уникальное учебное и справочное пособие по наиболее распространенным алгоритмам манипулирования данными, которые зарекомендовали себя как надежные и проверенные многими поколениями программистов. По данным журнала "Delphi Informant" за 2002 год, эта книга была признана сообществом разработчиков прикладных приложений на Delphi как «самая лучшая книга по практическому применению всех версий Delphi».В книге подробно рассматриваются базовые понятия алгоритмов и основополагающие структуры данных, алгоритмы сортировки, поиска, хеширования, синтаксического разбора, сжатия данных, а также многие другие темы, тесно связанные с прикладным программированием.


Питон — модули, пакеты, классы, экземпляры

Python - объектно-ориентированный язык сверхвысокого уровня. Python, в отличии от Java, не требует исключительно объектной ориентированности, но классы в Python так просто изучить и так удобно использовать, что даже новые и неискушенные пользователи быстро переходят на ОО-подход.


Как пасти котов. Наставление для программистов, руководящих другими программистами

«Как пасти котов» – это книга о лидерстве и руководстве, о том, как первое совмещать со вторым. Это, если хотите, словарь трудных случаев управления IT-проектами. Программист подобен кошке, которая гуляет сама по себе. Так уж исторически сложилось. Именно поэтому так непросто быть руководителем команды разработчиков. Даже если вы еще месяц назад были блестящим и дисциплинированным программистом и вдруг оказались в роли менеджера, вряд ли вы знаете, с чего надо начать, какой выбрать стиль руководства, как нанимать и увольнять сотрудников, проводить совещания, добиваться своевременного выполнения задач.