个人理解这个东西说白了就是当模板类(或函数)的类型参数为某特定值时用对应的特化定义代之。

看个例子吧

#include <iostream>

using namespace std;

template<typename T>
struct is_void {
static const bool value = false;
}; /* 上面的代码定义了一个简单的模板结构is_void的主版本,无论类型参数T是何值,
* 结构体的静态常量成员value的值都是false,这当然是无意义的,我们需要当且
* 仅当类型参数T为void时,value成员的值为true,于是我们定义下面的特化版本
*/ template<>
struct is_void<void> {
static const bool value = true;
}; /* 这里定义了一个is_void结构体的特化版本,将类型参数T固定为void,此时value
* 成员的值被设定为true
*/ int main(int argc, char* argv[]) {
cout << is_void<void>::value << endl;
cout << is_void<int>::value << endl;
cout << is_void<bool>::value << endl;
return ;
}

这段代码输出为:

1

0

0

以上代码中,当is_void结构体的模板类型参数T被指定时,编译器将检查该结构体的主版本及所有特化版本,如果类型参数与某一特化版本匹配,则该特化版本将被调用,如果没有匹配的特化版本,则调用主版本。也就是说如果T的实参为void,与我们定义的特化版本的参数一至,则此特化版本的定义被应用,如果T不是void则按主版本的定义展开。

上面的例子属于全特化(full-specialisation),也就是所有的类型参数都被确定为实际类型,但有时我们可以需要对参数进行部分而不是完全的限定,这便是偏特化(partial template-class specialisation)。

看下面的例子

template<typename T>
struct is_pointer {
static const bool value = false;
}; template<typename T>
struct is_pointer<T*> {
static const bool value = true;
};

这个例子中我们试图判断模板参数T是否是指针类型,如果是则value成员设为true,如果不是则置为false。然而无论T是不是指针都不可能被一一列举,也就是我们不能像第一个例子一样对诸如void*, int*, long*, char* .... 等等等等一一特化,这工作量太吓人了,我要需要一种方式来将指针的情况一次性特化,这就是例子中应用的方式:偏特化。偏特化的写法我想不难理解,但注意两个版本中的T是不一样的,我们可以把特化版本的T换成T2是没有问题的,这时对于is_pointer<int*>来讲,T代表int*,而T2代表的只是int,大家可以揣摩一下。这也就间接说明了主版本和特化版本允许不同个数的模板参数的原因。

再来一个怪异一点的例子

template <typename T>
struct remove_bounds {
typedef T type;
}; template <typename T, size_t N>
struct remove_bounds<T[N]> {
typedef T type;
}; remove_bounds<int>::type a;
remove_bounds<int[]>::type b;
remove_bounds<int[][]>::type c;

remove_bounds也就是将一个数组类型的[n]去掉即得到其基础类型,如果是非数组类型则返回原类型。于是上面的例子中,a是个int类型的变量,而b也是int类型,c有点怪,它不是int[5]类型,而是int[6]类型。这个例子也说明一个问题,特化版本与主版本的类型参数不一定一样,但typename修饰的类型参数个数是一致的。

C++ Template Specialization (模板特化)的更多相关文章

  1. C++ template —— 模板特化(五)

    本篇讲解模板特化-------------------------------------------------------------------------------------------- ...

  2. C++-函数模板特化如何避免重复定义

     我正在用一个基于模板的库源代码,该库包含一些针对特定类型的模板函数特化.类模板,函数模板和模板函数特化都在头文件中.我在我的.cpp文件中 #include 头文件并编译链接工程.但是为了在整个工程 ...

  3. C++模板之隐式实例化、显示实例化、隐式调用、显示调用和模板特化详解

    模板的实例化指函数模板(类模板)生成模板函数(模板类)的过程.对于函数模板而言,模板实例化之后,会生成一个真正的函数.而类模板经过实例化之后,只是完成了类的定义,模板类的成员函数需要到调用时才会被初始 ...

  4. C++ Primer 学习笔记_84_模板与泛型编程 --模板特化

    模板与泛型编程 --模板特化 引言: 我们并不总是能够写出对全部可能被实例化的类型都最合适的模板.某些情况下,通用模板定义对于某个类型可能是全然错误的,通用模板定义或许不能编译或者做错误的事情;另外一 ...

  5. C++ Primer 学习笔记_85_模板与泛型编程 --模板特化[续]

    模板与泛型编程 --模板特化[续] 三.特化成员而不特化类 除了特化整个模板之外,还能够仅仅特化push和pop成员.我们将特化push成员以复制字符数组,而且特化pop成员以释放该副本使用的内存: ...

  6. C++程序设计方法4:模板特化

    模板参数的具体化/特殊化 有时,有些类型不适用,则需要对模板进行特殊化处理,这称为“模板特化” 对函数模板,如果有多个模板参数,则特化时必须提供所有参数的特例类型,不能部分特化: 如: char *s ...

  7. C++ template —— 深入模板基础(二)

    上一篇C++ template —— 模板基础(一)讲解了有关C++模板的大多数概念,日常C++程序设计中所遇到的很多问题,都可以从这部分教程得到解答.本篇中我们深入语言特性.------------ ...

  8. C++ 模板特化以及Typelist的相关理解

    近日,在学习的过程中第一次接触到了Typelist的相关内容,比如Loki库有一本Modern C++ design的一本书,大概JD搜了一波没有译本,英文版600多R,瞬间从价值上看到了这本书的价值 ...

  9. 转:C++模板特化的概念

    http://blog.csdn.net/yesterday_record/article/details/7304025 很久没有看C++,在看STL源码剖析时,看到一个function templ ...

随机推荐

  1. Android MediaCodec 使用例子

    Android MediaCodec 使用例子 下面的例子是使用MediaCodec 录制到文件的例子. 1 public class AvcEncoder { private MediaCodec ...

  2. C#学习第三天

    经过这几天的学习,真的有点觉得以前C学的太不好现在学C#也不顺畅,虽然很多东西都似曾相识,但是就是还得看好几遍才能记得住,而且现在都是简单的东西,还没有看到重载等稍微难点的地方.应该好好努力了,昨天忙 ...

  3. .NET获取机器信息

    /// <summary> /// using System.Web; /// using System.Management; /// </summary> public s ...

  4. (转)PHP自定义遍历目录下所有文件dir(),readdir()函数

    方法一:使用dir()遍历目录 dir()函数,成功时返回Directory类实例 PHP dir() 语法格式为: dir(directory);//directory为需要显示文件名的目录名称,可 ...

  5. 在Cognos报表中使用钻取特性,参数传递

    转载至:http://blog.sina.com.cn/s/blog_6eda1c4e0100mu3t.html Cognos的钻取方式大致可以分为三种: 1.模型固有的->由CUBE和DMR支 ...

  6. 查看library_cache 库缓冲区的命中率

    关于library cache的命中率:    SQL> desc V$librarycache    NAMESPACE                                     ...

  7. HashMap、HashTable学习

    HashMap: HashMap 继承于AbstractMap,实现了Map.Cloneable.java.io.Serializable接口. HashMap 的实现不是同步的,这意味着它不是线程安 ...

  8. nodejs新手教程中upload file的问题

    可参见: http://cnodejs.org/topic/50234890f767cc9a51f88481 request.setEncoding("utf8");应注释掉.

  9. linux的du和df命令

    今天也有同学问我Linux下查看目录大小的命令,现在也将前阵子学习到du/df两个命令总结一下吧.前阵子测试工作中有遇到过由于磁盘空间满导致程序无法执行到情况,所以使用了df和du两个命令. du查看 ...

  10. PAT - IO-01. 表格输出(5)

    题目: 本题要求编写程序,按照规定格式输出表格. 输入格式: 本题目没有输入. 输出格式: 要求严格按照给出的格式输出下列表格: ----------------------------------- ...