Библиотечный тип remove_reference
был представлен в разделе 16.2.3, он работает с серией специализаций:
//
template
typedef T type;
};
//
//
template
{ typedef Т type; };
template
{ typedef T type; };
Первый шаблон определяет самую общую версию. Его экземпляр может быть создан с любым типом; он использует свой аргумент шаблона как тип для своего члена type
. Следующие два класса — это частичные специализации первоначального шаблона.
Поскольку частичная специализация — это шаблон, начнем, как обычно, с определения параметров шаблона. Подобно любой другой специализации, у частичной специализации то же имя, что и у специализируемого шаблона. Список параметров специализации шаблона включает элементы для каждого параметра шаблона, тип которого не был определен полностью при частичной специализации. После имени класса располагаются аргументы для параметров специализируемого шаблона. Эти аргументы располагаются в угловых скобках после имени шаблона. Аргументы позиционально соответствуют параметрам первоначального шаблона.
Список параметров шаблона частичной специализации — это подмножество или специализация списка параметров первоначального шаблона. В данном случае у специализаций то же количество параметров, что и у первоначального шаблона. Но тип параметров в специализациях отличается от первоначального шаблона. Специализация будут использоваться для ссылок на типы l- и r-значений соответственно:
int i;
//
remove_reference
//
//
remove_reference
//
//
remove_reference
У всех трех переменных, a
, b
и с
, тип int
.
Вместо специализации всего шаблона можно специализировать только одну или несколько его функций-членов. Например, если Foo
— это шаблон класса с членом Bar
, можно специализировать только этот член:
template
Foo (const T &t = T()): mem(t) { }
void Bar() { /* ... */ }
T mem;
//
};
template<> //
void Foo
{
//
//
}
Здесь специализируется только один член класса Foo
. Другие его члены предоставляются шаблоном Foo
:
Foo
fs.Bar(); //
Foo
fi.Bar(); //
При использовании шаблона Foo
с любым типом, кроме int
, члены экземпляра создаются, как обычно. При использовании шаблона Foo
с типом int
все члены экземпляра, кроме Bar
, создаются, как обычно. Если использовать член Bar
класса Foo
, то получится специализированное определение.
Упражнение 16.62. Определите собственную версию класса hash
и контейнер unordered_multiset
объектов класса Sales_data
. Поместите в контейнер несколько транзакций и выведите его содержимое.
Упражнение 16.63. Определите шаблон функции для подсчета количества вхождений заданного значения в векторе. Проверьте программу, передав ей вектор значений типа double
, вектор целых чисел и вектор строк.
Упражнение 16.64. Напишите специализированную версию шаблона из предыдущего упражнения для обработки вектора vector
и используйте ее в программе.