Симуляция частичной специализации - [3]

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

>TrueType const_discriminator(const volatile void*);

>FalseType const_discriminator(volatile void*);


>template‹class T›

>struct IsConst {

>private:

> static T t_;

>public:

> enum {value = sizeof(const_discriminator(&t_)) == sizeof(TrueType)};

>};


>template‹›

>class IsConst‹void› {

>public:

> enum {value = false};

>};


>TrueType volatile_discriminator(const volatile void*);

>FalseType volatile_discriminator(const void*);


>template‹class T›

>struct IsVolatile {

>private:

> static T t_;

>public>:

> enum {value = sizeof(volatile_discriminator(&t_)) – sizeof(TrueType)};

>};


>template‹›

>class IsVolatile‹void› {

>public:

> enum {value = false};

>};


Очевидно, что эти метафункции не работают, если в качестве аргумента им передан тип имеющий как const, так и volatile квалификацию. Реализация IsReference‹T› основывается на том факте, что добавление cv-квалификации к ссылке игнорируется:

>template‹class T›

>class IsReference {

>private:

> typedef T const volatile cv_t_;

>public:

> enum {value = !IsConst‹cv_t_›::value || !IsVolatile‹cv_t_›::value};

>};


>template‹› class IsReference‹void› {

>public:

> enum {value = false};

>};


Так как метафункция IsReference‹T› использует метафункции IsConst‹T› и IsVolatile‹T›, естественно, что она имеет те же недостатки.

ПРИМЕЧАНИЕ Описание и анализ других полезных метафункций, основанных на дискриминирующих функциях, выходит за рамки данной статьи и оставляется в качестве упражнения читателю. Например, можно построить метафункцию IsDerived‹T, Base›, позволяющую специализировать шаблоны для наследников определенного класса.

Еще одним достаточно важным ограничением техник симуляции частичной специализации является то, что еще никому не удавалось (и вряд ли удастся), например, получить тип T, имея T&. С использованием «настоящей» частичной специализации эта задача решается тривиально:

>template‹class T›

>struct RemoveReference {

> typedef T Type;

>};


>template‹class T›

>struct RemoveReference‹T&› {

> typedef T Type;

>};

Заключение

Описанная техника позволяет использовать преимущества частичной специализации шаблонов классов даже в случае отсутствия соответствующей поддержки со стороны компилятора. Комбинация приведенной методики с метафункциями при необходимости позволяет описывать достаточно сложные условия специализации шаблонов.

Единственным «серьезным» требованием к компилятору является наличие реализации шаблонов членов классов. Симуляция частичной специализации была проверена на следующих компиляторах:

•Microsoft Visual C++ 7.0 aka .NET

•Microsoft Visual C++ 6.0 SP4, SP5

•Intel C++ Compiler 4.0, 5.1, 6.0

•Borland C++ Command-line Compiler 5.51, 5.6

•GNU GCC 2.95.3-5

•Comeau C++ Compiler Online Version (compiled only)

Хотя последние четыре и поддерживают частичную специализацию, иногда может быть полезным прибегать к технике симуляции в случае одновременного использования нескольких компиляторов, один из которых «не дорос» до частичной специализации. При этом удобно, если использование условной компиляции можно минимизировать.

Комментарии:

>template‹class TRet, class TP1›

>class CDelegate1 {

> //…

>};


>template‹class TP1›

>class CDelegate1‹bool, TP1› {

> //…

>};


>template‹class TRet, class TP1, class TP2›

>class CDelegate2 {

> //…

>};


>template‹class TP1, class TP2›

>class CDelegate2‹bool, TP1, TP2› {

>//…

>};

и т.д…

Андрей 20.3.2003 12:22
... и статической T не надо

А мне как то больше понравился такой вариант (где нет статического T _t):

>template‹class T›

>class IsPointer {

>private:

> struct TrueType {> char dummy_ [1];> };

> struct FalseType { char dummy_ [2]; };

> struct PointerShim { PointerShim(const volatile void*); >};

> static TrueType ptr_discriminator(PointerShim);

> static FalseType ptr_discriminator(…);

> static T rett();

>public:

> enum {value = sizeof(ptr_discriminator(rett())) == sizeof(TrueType)};

>};


>template‹›

>class IsPointer‹void› {

>public:

> enum {value = false};

>};


Кстати, еще неплохо было бы дабавить IsArray, который таки почти смог добить Андрей Тарасевич в одном из топиков форума С++

PS Павел, кстати, эту же статью от вас я уже видел в каком то online издании… Или я ошибаюсь?

Andrew S 7.3.2003 17:50
А ссылку на boost.org?

Почему не указал ссылку на boost.org? Там уж намного больше готовых функций, чем ты привёл.

limax 7.3.2003 15:1


Рекомендуем почитать
Программирование на языке Пролог для искусственного интеллекта

Книга известного специалиста по программированию (Югославия), содержащая основы языка Пролог и его приложения для решения задач искусственного интеллекта. Изложение отличается методическими достоинствами — книга написана в хорошем стиле, живым языком. Книга дополняет имеющуюся на русском языке литературу по языку Пролог.Для программистов разной квалификации, специалистов по искусственному интеллекту, для всех изучающих программирование.


Программирование на Visual C++. Архив рассылки

РАССЫЛКА ЯВЛЯЕТСЯ ЧАСТЬЮ ПРОЕКТА RSDN, НА САЙТЕ КОТОРОГО ВСЕГДА МОЖНО НАЙТИ ВСЮ НЕОБХОДИМУЮ РАЗРАБОТЧИКУ ИНФОРМАЦИЮ, СТАТЬИ, ФОРУМЫ, РЕСУРСЫ, ПОЛНЫЙ АРХИВ ПРЕДЫДУЩИХ ВЫПУСКОВ РАССЫЛКИ И МНОГОЕ ДРУГОЕ.


Идиомы и стили С++

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


Человеческий фактор в программировании

Хорошее программное обеспечение создается людьми. Так же как и плохое. Именно поэтому основная тема этой книги — не аппаратное и не программное обеспечение, а человеческий фактор в программировании (peopleware). Первое издание «Constantine on Peopleware» признано классическим трудом в области информационных технологий. Новая книга Ларри Константина включает все 52 легендарные статьи из предыдущей книги и 25 новых эссе.Peopleware охватывает все аспекты, связанные с ролью людей в разработке программного обеспечения.


Единая система программной документации. Техническое задание. Требования к содержанию и оформлению

United system for program documentation. Technical specification for development. Requirements to contents and form of presentation Настоящий стандарт устанавливает порядок построения и оформления технического задания на разработку программы или программного изделия для вычислительных машин, комплексов и систем независимо от их назначения и области применения.Стандарт полностью соответствует СТ СЭВ 1627-79.Переиздание (Ноябрь 1987 г.) с Изменением № 1, утвержденным в июле 1981 г (ИУС 7-81)


Тонкости дизассемблирования

Очень часто под рукой не оказывается ни отладчика, ни дизассемблера, ни даже компилятора, чтобы набросать хотя бы примитивный трассировщик. Разумеется, что говорить о взломе современных защитных механизмов в таких условиях просто смешно, но что делать если жизнь заставляет?..