1. template<class T>
  2. class Compare
  3. {
  4. public:
  5. static bool isEqual(const T& lh,const T& rh)
  6. {
  7. return lh==rh;
  8. }
  9. };

这是一个用于比较的类模板,里面可以有多种用于比较的函数, 以IsEqual为例。

一、特化为绝对类型

也就是说直接为某个特定类型做特化,这是我们最常见的一种特化方式, 如特化为float, double等

  1. template<>
  2. class Compare<float>
  3. {
  4. public:
  5. static bool isEqual(const float &lh,const float& rh)
  6. {
  7. return abs(lh-rh)<10e-; //0.001
  8. }
  9. };

画红线的地方不可少。

  1. int main()
  2. {
  3. int a=,b=;
  4. cout<<Compare<int>::isEqual(a,b)<<endl;
  5. float x=3.141,y=3.1415; //如果不要特化就不相等,用特化就相等
  6. cout<<Compare<float>::isEqual(x,y)<<endl;
  7. }

二、特化为引用,指针类型

这种特化我最初是在stl源码的的iterator_traits特化中发现的, 如下:

  1. template <class _Iterator>
  2. struct iterator_traits {
  3. typedef typename _Iterator::iterator_category iterator_category;
  4. typedef typename _Iterator::value_type value_type;
  5. typedef typename _Iterator::difference_type difference_type;
  6. typedef typename _Iterator::pointer pointer;
  7. typedef typename _Iterator::reference reference;
  8. };
  9.  
  10. // specialize for _Tp*
  11. template <class _Tp>
  12. struct iterator_traits<_Tp*> {
  13. typedef random_access_iterator_tag iterator_category;
  14. typedef _Tp value_type;
  15. typedef ptrdiff_t difference_type;
  16. typedef _Tp* pointer;
  17. typedef _Tp& reference;
  18. };

当然,除了T*, 我们也可以将T特化为 const T*, T&, const T&等,以下还是以T*为例:

  1. #include<iostream>
  2. using namespace std;
  3.  
  4. template<class T>
  5. class Compare
  6. {
  7. public:
  8. static bool isEqual(const T& lh,const T& rh)
  9. {
  10. return lh==rh;
  11. }
  12. };
  13.  
  14. template< class T>
  15. class Compare<T*>
  16. {
  17. public:
  18. static bool isEqual(const T* lh,const T* rh)
  19. {
  20. cout<<"这是调用了Compare<T*>类型"<<endl;
  21. return Compare<T>::isEqual(*lh,*rh);
  22. }
  23. };
  24.  
  25. int main()
  26. {
  27. int a=,b=;
  28. cout<<Compare<int>::isEqual(a,b)<<endl;
  29.  
  30. cout<<Compare<int*>::isEqual(&a,&b)<<endl;//调用指针类型
  31. }

输出:

1
这是调用了Compare<T*>类型
1
请按任意键继续. . .

后面的由于传入的是Compare<int*>类型,传入的指针类型,调用特化的指针类型函数。

这种特化其实是就不是一种绝对的特化, 它只是对类型做了某些限定,但仍然保留了其一定的模板性,这种特化给我们提供了极大的方便, 如这里, 我们就不需要对int*, float*, double*等等类型分别做特化了。

三、特化为另外一个类模板

这其实是第二种方式的扩展,其实也是对类型做了某种限定,而不是绝对化为某个具体类型,如下:

  1. #include<iostream>
  2. #include<vector>
  3. using namespace std;
  4.  
  5. template<class T>
  6. class Compare
  7. {
  8. public:
  9. static bool isEqual(const T& lh,const T& rh)
  10. {
  11. return lh==rh;
  12. }
  13. };
  14.  
  15. //specialize for float
  16.  
  17. template<>
  18. class Compare<float>
  19. {
  20. public:
  21. static bool isEqual(const float &lh,const float& rh)
  22. {
  23. return abs(lh-rh)<10e-; //0.001
  24. }
  25. };
  26. template< class T>
  27. class Compare<vector<T> >
  28. {
  29. public:
  30. static bool isEqual(const vector<T> &lh,const vector<T>& rh)
  31. {
  32. cout<<"调用了compare<vector<T> >"<<endl;
  33. if(lh.size()!=rh.size()) return false;
  34. else
  35. {
  36. for(int i=;i<lh.size();i++)
  37. {
  38. if(lh[i]!=rh[i])
  39. return false;
  40. }
  41. }
  42. return true;
  43. }
  44. };
  45. int main()
  46. {
  47. int a=,b=;
  48. cout<<Compare<int>::isEqual(a,b)<<endl;
  49.  
  50. vector<int> x(,);
  51. vector<int> y(,);
  52. cout<<Compare<vector<int> >::isEqual(x,y);
  53. }

这就把IsEqual的参数限定为一种vector类型, 但具体是vector<int>还是vector<float>, 我们可以不关心, 因为对于这两种类型,我们的处理方式是一样的,我们可以把这种方式称为“半特化”。

当然, 我们可以将其“半特化”为任何我们自定义的模板类类型:

  1. #include<iostream>
  2. #include<vector>
  3. using namespace std;
  4.  
  5. template<class T>
  6. class Compare
  7. {
  8. public:
  9. static bool isEqual(const T& lh,const T& rh)
  10. {
  11. return lh==rh;
  12. }
  13. };
  14.  
  15. //specialize for float
  16.  
  17. template<>
  18. class Compare<float>
  19. {
  20. public:
  21. static bool isEqual(const float &lh,const float& rh)
  22. {
  23. return abs(lh-rh)<10e-; //0.001
  24. }
  25. };
  26.  
  27. //specialize for any template class Type
  28. template<class T1>
  29. struct SpecializedType
  30. {
  31. T1 x1;
  32. T1 x2;
  33. };
  34. template<class T>
  35. class Compare<SpecializedType<T> >
  36. {
  37. public:
  38. static bool isEqual(const SpecializedType<T> & lh,const SpecializedType<T> & rh)
  39. {
  40. cout<<"使用了Compare<SpecailizedType<T> >"<<endl;
  41. return Compare<T>::isEqual(lh.x1+lh.x2,rh.x1+rh.x2);
  42. }
  43. };
  44. int main()
  45. {
  46.  
  47. SpecializedType<int> s1={,};
  48. SpecializedType<int> s2={,};
  49. cout<<Compare<SpecializedType<int> >::isEqual(s1,s2)<<endl;
  50. }

使用了Compare<SpecailizedType<T> >
1
请按任意键继续. . .

注意在

class Compare<SpecializedType<T> >
我们Compare传入的类型是SpecializedType<int> >,所以比较的类型是

SpecializedType<T> ,而不是T。
当然,我们还可以:

// custom template class
SpecializedType<float> s1 = {10.1f,10.2f};
SpecializedType<float> s2 = {10.3f,10.0f};
bool r6 = Compare<SpecializedType<float> >::IsEqual(s1, s2);
使用特化的类型float。

模板有两种特化,全特化和偏特化(局部特化)

模板函数只能全特化,没有偏特化(以后可能有)。

模板类是可以全特化和偏特化的。

全特化,就是模板中模板参数全被指定为确定的类型。

全特化也就是定义了一个全新的类型,全特化的类中的函数可以与模板类不一样。

偏特化,就是模板中的模板参数没有被全部确定,需要编译器在编译时进行确定。

在类型上加上const、&、*( cosnt int、int&、int*、等等)并没有产生新的类型。只是类型被修饰了。模板在编译时,可以得到这些修饰信息。

模板的特化是非常有用的。它像一个在编译期的条件判断。当编译器在编译时找到了符合的特化实现,就会使用这个特化实现。这就叫编译器多态(或者叫静态多态)。这种东西对编写基础库是很有用的。这也就是为何c++的基础库大量使用了模板技术,而且大量使用了特化,特别是偏特化。

在泛型中,利用特化类得到类新的特性,以便找到最适合这种特性的实现。而这一切都是在编译时完成。

  1. 转自:http://www.360doc.com/content/12/1210/12/9200790_253182813.shtml#

STL 全特化/偏特化的更多相关文章

  1. c++模板特化偏特化

    模板为什么要特化,因为编译器认为,对于特定的类型,如果你对某一功能有更好地实现,那么就该听你的. 模板分为类模板与函数模板,特化分为全特化与偏特化.全特化就是限定死模板实现的具体类型,偏特化就是模板如 ...

  2. STL全特化与偏特化

    在泛型编程中,常常会使用一些非完全泛型的类模板,这就是特化. 如何理解全特化呢?如上图所示,第一个template class是空间配置器的类模板,第二个就是一个全特化的template class. ...

  3. C++的模板特化 和 STL中iterator_traits模板的偏特化

    C++中有类模板和函数模板,它们的定义如下所示: 类模板: template<class T1,class T2> class C { //... }; 函数模板: template< ...

  4. STL迭代器之一:偏特化

    在stl的算法中运用容器的迭代器时,很可能经常会用到迭代器相应型别(例如迭代器所指物的型别),假设算法中有必要声明一个变量,以"迭代器所指对象的型别"为类型,如何是好,例如我们写一 ...

  5. C++模板编程里的主版本模板类、全特化、偏特化(C++ Type Traits)

    1.  主版本模板类 首先我们来看一段初学者都能看懂,应用了模板的程序: 1 #include <iostream> 2 using namespace std; 3 4 template ...

  6. [C++ 2011 STL (VS2012 Update4) 源代码阅读系列(2)]熟悉一些宏定义和模版偏特化或叫模版专门化

    [C++ 2011 STL (VS2012 Update4) 源代码阅读系列(2)]熟悉一些宏定义和模版偏特化或叫模版专门化 // point_test.cpp : 知识点练习和测试,用于单步调试,跟 ...

  7. C++ 模板的全特化与偏特化

    模板为什么要特化,因为编译器认为,对于特定的类型,如果你能对某一功能更好的实现,那么就该听你的. 模板分为类模板与函数模板,特化分为全特化与偏特化.全特化就是限定死模板实现的具体类型,偏特化就是如果这 ...

  8. C++模板的偏特化与全特化

    模板的声明 类模板和函数模板的声明方式是一样的,在类定义/模板定义之前声明模板参数列表.例如: // 类模板 template <typename T1, typename T2> cla ...

  9. C++模板函数只能全特化不能偏特化

    C++模板函数只能全特化不能偏特化

随机推荐

  1. stat 函数讲解

    表头文件:    #include <sys/stat.h>                     #include <unistd.h>定义函数:    int stat( ...

  2. 使用runOnUiThread更新UI

    android中更新UI的方式比较多,这里就不一一介绍了,比较常用的Thread+Handler,但是这种方式较繁琐,如在使用ProgressDialog创建进度对话框一文中就是使用的这种方式更新UI ...

  3. Linux网络管理——子网掩码

    1. 网络基础 .note-content {font-family: "Helvetica Neue",Arial,"Hiragino Sans GB",&q ...

  4. artTemplate-3.0

    https://github.com/aui/artTemplate artTemplate-3.0 新一代 javascript 模板引擎 目录 特性 快速上手 模板语法 下载 方法 NodeJS ...

  5. git/github 笔记

    2016-1-9 创建github repos并提交修改 在[这里](https://github.com/new)创建一个repos, 进入终端,cd到一个目录下,这个目录用来放等下clone的工程 ...

  6. 基于Socket的UDP和TCP编程介绍

    一.概述 TCP(传输控制协议)和UDP(用户数据报协议是网络体系结构TCP/IP模型中传输层一层中的两个不同的通信协议. TCP:传输控制协议,一种面向连接的协议,给用户进程提供可靠的全双工的字节流 ...

  7. 修改 tomcat 内存

    在Jetty 的VM参数中设置: -Xms256m -Xmx512m -XX:MaxNewSize=256m -XX:MaxPermSize=256m 在tomcat运行环境中设置: window环境 ...

  8. java.sql.ResultSetMetaData.getColumnLabel和getColumnName的区别

    如果将ResultSet的结果映射到HashMap中,要使用getColumnLabel,而不要用getColumnName,这样可提高程序的健壮性 理由: getColumnName返回的是sql语 ...

  9. Nginx模块开发入门(转)

    前言 Nginx是当前最流行的HTTP Server之一,根据W3Techs的统计,目前世界排名(根据Alexa)前100万的网站中,Nginx的占有率为6.8%.与Apache相比,Nginx在高并 ...

  10. Quality Over Quantity: 更少一些,更好一些_第1页_福布斯中文网

    Quality Over Quantity: 更少一些,更好一些_第1页_福布斯中文网     Quality Over Quantity: 更少一些,更好一些    2013年04月09日     ...