c++ template之trait技法总结

Trait技法的最初目的是为了管理模板参数,有的时候模板中需要几个参数,但是往往有些参数是与main parameters紧密相关的,这时候可以使用trait技巧,从几个主要的模板参数中推导出相应的secondary template argument,并以默认模板参数的形式出现在模板中。实际上使用的trait技巧实例往往会有效地提高程序的效率,下面我结合STL小小的说说trait的实际运用。

1、SGI STL中的__type_traits

类型粗略的讲有两种,一种是用class封转,并且它的复制拷贝很费时,比如需要深度复制,需要safe copying——即需要调用构造函数完成复制;另一种是所谓的POD(plain old data),一般是build-in类型或是c式的struct类型,这种类型复制和拷贝时只需直接复制内存块就可以了,于是有了bitwise copying或是trival construct/deconstruct/copy/assignment的概念。

SGI STL中使用__type_traits就可以在编译期间就确定复制一个对象是调用赋值构造函数还是直接调用memcpy,从而提高效率。看SGI STL的源代码:

  1. struct __true_type {};
  2. struct __false_type {};
  3. template <class type>//primariy template
  4. struct __type_traits {
  5. typedef __true_type this_dummy_member_must_be_first;
  6. typedef __false_type has_trivial_default_constructor;
  7. typedef __false_type has_trivial_copy_constructor;
  8. typedef __false_type has_trivial_assignment_operator;
  9. typedef __false_type has_trivial_destructor;
  10. typedef __false_type is_POD_type;
  11. };
  12. __STL_TEMPLATE_NULL struct __type_traits<char> {//full specialization
  13. typedef __true_type has_trivial_default_constructor;
  14. typedef __true_type has_trivial_copy_constructor;
  15. typedef __true_type has_trivial_assignment_operator;
  16. typedef __true_type has_trivial_destructor;
  17. typedef __true_type is_POD_type;
  18. };
  19. __STL_TEMPLATE_NULL struct __type_traits<int> {
  20. typedef __true_type has_trivial_default_constructor;
  21. typedef __true_type has_trivial_copy_constructor;
  22. typedef __true_type has_trivial_assignment_operator;
  23. typedef __true_type has_trivial_destructor;
  24. typedef __true_type is_POD_type;
  25. };
struct __true_type {};
struct __false_type {};

template <class type>   //primariy template
struct __type_traits { 
   typedef __true_type     this_dummy_member_must_be_first;
    
   typedef __false_type    has_trivial_default_constructor;
   typedef __false_type    has_trivial_copy_constructor;
   typedef __false_type    has_trivial_assignment_operator;
   typedef __false_type    has_trivial_destructor;
   typedef __false_type    is_POD_type;
};
__STL_TEMPLATE_NULL struct __type_traits<char> { //full specialization
   typedef __true_type    has_trivial_default_constructor;
   typedef __true_type    has_trivial_copy_constructor;
   typedef __true_type    has_trivial_assignment_operator;
   typedef __true_type    has_trivial_destructor;
   typedef __true_type    is_POD_type;
};
__STL_TEMPLATE_NULL struct __type_traits<int> {
   typedef __true_type    has_trivial_default_constructor;
   typedef __true_type    has_trivial_copy_constructor;
   typedef __true_type    has_trivial_assignment_operator;
   typedef __true_type    has_trivial_destructor;
   typedef __true_type    is_POD_type;
};
  1. ......//针对其他build-in类型的特化
......//针对其他build-in类型的特化
接下来,看看怎样使用__type_trairt:

  1. template <class InputIterator,class ForwardIterator, class T>
  2. inline ForwardIterator __uninitialized_copy(InputIterator first, InputIterator last,
  3. ForwardIterator result, T*) {
  4. typedef typename __type_traits<T>::is_POD_type is_POD;
  5. return __uninitialized_copy_aux(first, last, result, is_POD());
  6. }
  7. template <class InputIterator,class ForwardIterator> inline ForwardIterator
  8. __uninitialized_copy_aux(InputIterator first, InputIterator last,
  9. ForwardIterator result, __true_type) {
  10. return copy(first, last, result); //STL 算法copy里面大有乾坤,针对POD类型,调用memmove,还含有iterator_trait的使用
  11. }
  12. template <class InputIterator,class ForwardIterator>
  13. ForwardIterator __uninitialized_copy_aux(InputIterator first, InputIterator last,
  14. ForwardIterator result, __false_type) {
  15. ForwardIterator cur = result;
  16. for ( ; first != last; ++first, ++cur)
  17. construct(&*cur, *first);
  18. return cur;
  19. }
  20. }
template <class InputIterator, class ForwardIterator, class T>
inline ForwardIterator __uninitialized_copy(InputIterator first, InputIterator last,
                     ForwardIterator result, T*) {
  typedef typename __type_traits<T>::is_POD_type is_POD;
  return __uninitialized_copy_aux(first, last, result, is_POD());
}
template <class InputIterator, class ForwardIterator> inline ForwardIterator 
__uninitialized_copy_aux(InputIterator first, InputIterator last,
                         ForwardIterator result, __true_type) {
  return copy(first, last, result); //STL 算法copy里面大有乾坤,针对POD类型,调用memmove,还含有iterator_trait的使用
}
template <class InputIterator, class ForwardIterator>
ForwardIterator __uninitialized_copy_aux(InputIterator first, InputIterator last,
                         ForwardIterator result, __false_type) {
  ForwardIterator cur = result;
    for ( ; first != last; ++first, ++cur)
      construct(&*cur, *first);
    return cur;
  }
}
其他的如uninitialized_fill、 uninitialized_fill_n也是类似的。还有《C++ template》中的CSMTraits也很能说明trait的强大之处。

2、STL中的iterator_traits

iterator_traits定义了迭代器的value type、different type、reference type、pointer type和iterator_category类型,我们只说最后一个。

  1. struct input_iterator_tag {};
  2. struct output_iterator_tag {};
  3. struct forward_iterator_tag :public input_iterator_tag {};
  4. struct bidirectional_iterator_tag : public forward_iterator_tag {};
  5. struct random_access_iterator_tag :public bidirectional_iterator_tag {};
struct input_iterator_tag {};
struct output_iterator_tag {};
struct forward_iterator_tag : public input_iterator_tag {};
struct bidirectional_iterator_tag : public forward_iterator_tag {};
struct random_access_iterator_tag : public bidirectional_iterator_tag {};
  1. template <class T,class Distance> struct input_iterator {
  2. typedef input_iterator_tag iterator_category;
  3. };
  4. struct output_iterator {
  5. typedef output_iterator_tag iterator_category;
  6. };
  7. template <class T,class Distance> struct forward_iterator {
  8. typedef forward_iterator_tag iterator_category;
  9. };
  10. template <class T,class Distance> struct bidirectional_iterator {
  11. typedef bidirectional_iterator_tag iterator_category;
  12. };
  13. template <class T,class Distance> struct random_access_iterator {
  14. typedef random_access_iterator_tag iterator_category;
  15. };
template <class T, class Distance> struct input_iterator {
  typedef input_iterator_tag iterator_category;
};
struct output_iterator {
  typedef output_iterator_tag iterator_category;
};
template <class T, class Distance> struct forward_iterator {
  typedef forward_iterator_tag iterator_category;
};
template <class T, class Distance> struct bidirectional_iterator {
  typedef bidirectional_iterator_tag iterator_category;
};
template <class T, class Distance> struct random_access_iterator {
  typedef random_access_iterator_tag iterator_category;
};
  1. //iterator_traits的使用
//iterator_traits的使用
  1. <pre class="cpp" name="code">template <class InputIterator, class Distance>
  2. inline void advance(InputIterator& i, Distance n) {
  3. __advance(i, n, iterator_traits<InputIterator>::iterator_category());//构造一个临时的xx_iterator_tag对象
  4. }
  5. template <class InputIterator,class Distance>
  6. inline void __advance(InputIterator& i, Distance n, input_iterator_tag) {
  7. while (n--) ++i;
  8. }
  9. template <class BidirectionalIterator,class Distance>
  10. inline void __advance(BidirectionalIterator& i, Distance n,
  11. bidirectional_iterator_tag) {
  12. if (n >= 0)
  13. while (n--) ++i;
  14. else
  15. while (n++) --i;
  16. }
  17. template <class RandomAccessIterator,class Distance>
  18. inline void __advance(RandomAccessIterator& i, Distance n,
  19. random_access_iterator_tag) {
  20. i += n;
  21. }</pre>
  22. <pre></pre>
  23. <p></p>
  24. <pre></pre>
  25. <p></p>
  26. <p><span style="font-size: 18px;"><span style="white-space: pre;"></span>3、STL中的char_traits</span></p>
  27. <p><span style="white-space: pre;"><span style="font-size: 18px;"></span></span></p>
  28. <p><span style="font-size: 18px;"><span style="white-space: pre;"></span>与trait相关的还有policy技巧,trait更偏向于从主模板参数推导出与之紧密相关的类型或常量等信息,而policy更偏向于行为方面的信息,如函数。trait与policy的界限有些模糊,实际上有的书也有property trait和policy trairt的概念,由此也可以看出,policy类似于定义的一种算法(函数),而与主模板参数没有很紧密的逻辑联系。</span></p>
  29. <p><span style="font-size: 18px;"><br>
  30. </span></p>
  31. <p><span style="font-size: 18px;">参考书籍:《STL源码剖析》</span></p>
  32. <p><span style="font-size: 18px;"><span style="white-space: pre;"></span> 《C++template》</span></p>
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值