模板为什么要特化,因为编译器认为,对于特定的类型,如果你能对某一功能更好的实现,那么就该听你的。
模板分为类模板与函数模板,特化分为全特化与偏特化。全特化就是限定死模板实现的具体类型,偏特化就是如果这个模板有多个类型,那么只限定其中的一部分。
template<typename T1, typename T2>
class Test
{
public:
Test(T1 i,T2 j):a(i),b(j){cout<<"模板类"<<endl;}
private:
T1 a;
T2 b;
};
template<>
class Test<int , char>
{
public:
Test(int i, char j):a(i),b(j){cout<<"全特化"<<endl;}
private:
int a;
char b;
};
template <typename T2>
class Test<char, T2>
{
public:
Test(char i, T2 j):a(i),b(j){cout<<"偏特化"<<endl;}
private:
char a;
T2 b;
};
那么下面3句依次调用类模板、全特化与偏特化:
Test<double , double> t1(0.1,0.2);
Test<int , char> t2(1,'A');
Test<char, bool> t3('A',true);
//依次打印:
//类模板
//全特化
//偏特化
另外函数模板只能全特化不能偏特化
从上面的例子可以看出:
全特化,模板的参数列表为空,表示该特化版本没有模板参数,全部被特化了。
偏特化,模板参数列表中包含未特化的模板参数,同时指定特化额类型,比如上述中char为T1的特化类型。
想起iterator_trait中对原生指针和const原生指针的特化版本:
template <typename T>
struct iterator_traits<T*>//为原生指针设计的特化版本
{
typedef T value_type;
};
template <typename T>
struct iterator_traits<const T*>//为const 指针设计的特化版本
{
typedef T value_type;
};
这里是一个偏特化的用法。指定了当迭代器类型为T*或者constT*这种原生指针时的行为。