__type_traits:双底线是说明这是SGI STL内部使用的东西,不在STL标准范围之内。iterator_traits负责萃取迭代器(iterator)的特性。而__type_traits则负责萃取型别(type)的特性。
我们所关注的型别特性是指:这个型别是否具备non-trivial defalt ctor,non-trivial copy cotr ,non-trivial assignment operator,non-trivial dtor[1] ,如果没有这些non-trivial(有意义)函数,那么我们在对这个型别进行构造、析构、拷贝、赋值操作时,就可以采用最效的措施(根本不调用那些身居高位的constructor,destructor等,而采用内存直接处理操作如malloc(),memcpy()等,获得最高效率)。
一、那么什么是trivial or non-trivial呢?
这个trivial和non-trivial是对类的四种函数来说的:
- 构造函数(ctor)
- 拷贝构造函数(copy)
- 赋值函数(assignment)
- 析构函数(dtor)
如果至少满足下面3条里的一条:
- 显式(explict)定义了这四种函数。
- 类里有非静态非POD的数据成员。
- 有基类。
那么上面的四种函数是non-trivial函数,比如non-trivial ctor、non-trivial copy…POD意思是Plain Old Data(普通旧数据),也就是C++的内建类型或传统的C结构体类型,POD类型必然有(trivial:无意义)trivial ctor/dtor/copy/assignment四种函数。
二、__type_traits提供了一种机制:就是在编译时观察这个型别有没有trivial函数,如果有,那么就能执行高效率的操作了,哇好开心
当我们准备对一个“元素型别未知”的数组执行copy操作时,如果我们能事先知道其元素性别是否有一个trivail copy contructor,那么就能够帮助我们是否可以使用快速的memcpy()或memmove()。即在编译时期针对不同型别完成函数派送。
我们希望程序之中可以这样运用__type_traits<T>:
1 __type_traits<T>::has_trivial_default_constructor; 2 __type_traits<T>::has_trivial_copy_constructor; 3 __type_traits<T>::has_trivial_assignment_operator; 4 __type_traits<T>::has_trivial_destructor; 5 __type_traits<T>::is_POD_type;
我们希望利用响应结果来进行参数推导,而编译器只有面对class object形式的参数才会做参数推导,所以上述的返回结果应该如下:这两个classes没有任何成员,不会带来额外负担,却又能够标识真假。
1 struct __true_type {};
2 struct __false_type {};
为了达成上述五个式子,__type_traits内必须定义一些typedefs,其值不是__true_type就是__false_type:
1 1 template <class type> 2 2 struct __type_traits { 3 3 typedef __true_type this_dummy_member_must_be_first; //为了确保万一编译器也使用一个名为__type_traits而其实与此处定义并无任何关联的template时,事情顺利工作。 4 5 4 typedef __false_type has_trivial_default_constructor; 6 5 typedef __false_type has_trivial_copy_constructor; 7 6 typedef __false_type has_trivial_assignment_operator; 8 7 typedef __false_type has_trivial_destructor; 9 8 typedef __false_type is_POD_type; 10 9 };
针对__false_type,由于编译器有可能为各型别自动产生专属的__type_traits特化版本,所以你可以将上述成员次序重新排列,也可以移除上述任何成员,绝对不可以在没有改变编译器对应名称的情况下对成员重新命名,只有在编译器中加上适当支持时,否则新加入成员会被视为一般成员。
SGI将所有的内嵌型别定义成__false_type,因为SGI先定义最保守的值,然后再针对每一个标量型别(scalar types)设计适当的__type_traits特化版本:
# 什么是标量型别:标量类型(Scalar type)是相对复合类型(Compound type)来说的:标量类型只能有一个值,而复合类型可以包含多个。
上述__type_traits可以接受任何型别的参数,五个typedefs将由以下管道获得实值:
- 一般具现体(general instantiation),内含对所有型别都有效的保守值,上述各个has_trivial_xxx型别都被定义为_false_type,就是对所有型别都必定有效的保守值。
# 实例化( Instantiation) 与特化( Specialization),在 template 中,「以实际值( actual values)做为 template arguments,从而产生常规的( regular)class、 function 或 member function」,这个过程称为 「 template 实例化」( template instantiation)。
这 些 随 之 产 生 的 物 体(entity ;包 括 class, function 或 member function ) 通 称为 特 化 体 ( specialization)。( 译注:我所阅读的众多泛型编程书籍中,很少将这些物体称为特化体,较 常称呼的是具现体, instantiation)
- 经过声明的特化版本,例如<type_traits.h>内对所有C++标量型别(scalar types)提供了对应的特化版本声明(每个型别的值都是__true_type)。
- 某些编译器(如Silicon Graphics N32和N64编译器)会自动为所有型别提供适当的特化版本(这个是真的牛逼,不过到底有没有候捷对它表示疑惑)
三、对所有C++标量型别所定义的__type_traits的特化版本,非常必要
针对C++基本型别char,signed char ,unsigend char ,short,unsigned short ,int ,unsigned int,long,unsigned long,float,double,long double提供特化版本,每个型别的值都是__true_type,表示这些型别都可以采用最快速方式来进行拷贝或赋值操作。
__STL_TEMPLATE_NULL定义为template<>即class template explicit specialization(类模板显式特化)
1 //针对char的特化版本 2 __STL_TEMPLATE_NULL struct __type_traits<char> { 3 typedef __true_type has_trivial_default_constructor; 4 typedef __true_type has_trivial_copy_constructor; 5 typedef __true_type has_trivial_assignment_operator; 6 typedef __true_type has_trivial_destructor; 7 typedef __true_type is_POD_type; 8 }; 9 //针对signed char的特化版本 10 __STL_TEMPLATE_NULL struct __type_traits<signed char> { 11 typedef __true_type has_trivial_default_constructor; 12 typedef __true_type has_trivial_copy_constructor; 13 typedef __true_type has_trivial_assignment_operator; 14 typedef __true_type has_trivial_destructor; 15 typedef __true_type is_POD_type; 16 }; 17 //针对unsigned char的特化版本 18 __STL_TEMPLATE_NULL struct __type_traits<unsigned char> { 19 typedef __true_type has_trivial_default_constructor; 20 typedef __true_type has_trivial_copy_constructor; 21 typedef __true_type has_trivial_assignment_operator; 22 typedef __true_type has_trivial_destructor; 23 typedef __true_type is_POD_type; 24 }; 25 //针对short的特化版本 26 __STL_TEMPLATE_NULL struct __type_traits<short> { 27 typedef __true_type has_trivial_default_constructor; 28 typedef __true_type has_trivial_copy_constructor; 29 typedef __true_type has_trivial_assignment_operator; 30 typedef __true_type has_trivial_destructor; 31 typedef __true_type is_POD_type; 32 }; 33 ...未全部给出
注意:还有针对原生指针涉设计的__type_traits偏特化版本:原生指针也被视为一种标量型别。
1 template <class T> 2 struct__type_traits<T*> { 3 typedef __true_type has_trivial_default_constructor; 4 typedef __true_type has_trivial_copy_constructor; 5 typedef __true_type has_trivial_assignment_operator; 6 typedef __true_type has_trivial_destructor; 7 typedef __true_type is_POD_type; 8 };
四、__type_traits的应用
1.uninitialized_fill_n()全局函数
1 template <class ForwardIterator, class Size, class T> 2 inline ForwardIteratoruninitialized_fill_n(ForwardIterator first, Size n,const T& x) { 4 return __uninitialized_fill_n(first, n, x, value_type(first)); 5 }
该函数以x为蓝本,自迭代器first开始构造n个元素。为求取最大效率,首先以value_type()萃取出迭代器first的 value type,再利用__type_traits判断型别是否为POD类型:
1 template <class ForwardIterator, class Size, class T, class T1> 2 inline ForwardIterator__uninitialized_fill_n(ForwardIterator first, Size n,const T& x, T1*) { 4 typedef typename __type_traits<T1>::is_POD_type is_POD; 5 return __uninitialized_fill_n_aux(first, n, x, is_POD()); 7 }
对于“是否是POD型别”采取最适当的措施:
1 //如果是POD型别,就会派送到这里。如果copy constructor等同于assignment,而且有trivial destructor。以下就有效: 2 template <class ForwardIterator, class Size, class T> 3 inline ForwardIterator 4 __uninitialized_fill_n_aux(ForwardIteratorfirst, Size n, 5 const T& x,__true_type) { 6 return fill_n(first, n, x); //交由高阶函数执行,如下所示 7 } 8 //以下是定义于<stl_algobase.h>中的fill_n 9 template <class OutputIterator, class Size, class T> 10 OutputIteratorfill_n(OutputIterator first, Size n, const T& value) { 11 for ( ; n > 0; --n, ++first) 12 *first = value; 13 return first; 14 } 15 16 //如果不是POD型别:就需要老老实实的一个一个在未初始化的内存中调用构造函数了!好伤心! 17 template <class ForwardIterator, class Size, class T> 18 ForwardIterator 19 __uninitialized_fill_n_aux(ForwardIteratorfirst, Size n, 20 const T& x,__false_type) { 21 ForwardIterator cur = first; 22 for ( ; n > 0; --n, ++cur) 23 construct(&*cur, x); 24 return cur; 25 }
2.destroy()函数
3.copy()全局函数
这个函数有非常多的特化(speialization)与强化(refinement)版本,殚精竭虑都是为了效率考虑,希望在最适当的情况下次采用最“雷霆万钧”的手段:
1 //拷贝一个数组,其元素性别为任意性别,根据情况采用最有效率的拷贝手段 2 template <class T>inline void copy(T* source,T* destination,int n){ 3 copy(source,destination,n,typename __type_traits<T>::has_trivial_copy_constructor()); 4 } 5 6 //拷贝一个数组,其元素性别拥有non-trivial copy constructors(有意义拷贝构造函数) 7 template <class T>void copy(T* source,T* destination,int __false_type) 8 {...} 9 10 //拷贝一个数组,其元素型别拥有trivial copy constructors 11 //可借助memcpy()完成工作 12 template <class T>void copy(T* source,T* destination,int ,__true_type) 13 {...}
五、自己定义类型的__type_traits
假设自行定义了一个Shape class,除了某些厉害的编译器(如Silicon Graphics N32和N64编译器),对于一般编译器,__type_traits针对Shape萃取出来的每个特性都是__false_type,即使Shape是个POD类型。这样虽然过于保守,但是别无选择,除非针对Shape,自行设计一个__type_traits版本:
1 template<> struct __type_traits<Shape> { 2 typedef __false_type has_trivial_default_constructor; 3 typedef __true_type has_trivial_copy_constructor; 4 typedef __true_type has_trivial_assignment_operator; 5 typedef __true_type has_trivial_destructor; 6 typedef __false_type is_POD_type; 7 };
究竟什么时候一个class该有自己non-trivial_xxx,即如果class内含指针成员,并且对它进行内存动态配置,那么这个class就需要实现自己的non-trivial-xxx。
六、总结
有了__type_traits,即使你不能针对自己定义的型别,设计__type_traits特化版本,但是对于C++标量型别,我们可以采用最有效的拷贝或赋值操作,因为每个标量型别都有对应的__type_traits版本,其中每个typedef的值都是__true_type。
七、参考文献
[1].https://freezhongzi.info/index.php/2012/11/09/c-trivial%e5%92%8cnon-trivial%e6%9e%84%e9%80%a0%e5%87%bd%e6%95%b0%e5%8f%8apod%e7%b1%bb%e5%9e%8b/