如何正确使用enable_if?

时间:2017-01-09 16:33:21

标签: c++ c++11 iterator enable-if

我需要学习如何使用enable_if。为此,我需要使用enable_if重新实现distance函数。我试过这个:

#include <iostream>
#include <vector>
#include <list>
#include <utility>
#include <type_traits>

template<class In>
typename std::enable_if<!std::is_random_acces_iterator<In>::value, std::iterator_traits<In>::difference_type>::type  my_distance(In begin, In end, std::input_iterator_tag dummy){
  typename std::iterator_traits<In>::difference_type n = 0;
  while(begin!=end){
    ++begin; ++n;
  }
  std::cout << "STEPPING" << std::endl;
  return n;
}

template<class Ran>
typename std::enable_if<std::is_random_acces_iterator<Ran>::value, std::iterator_traits<In>::difference_type>::type my_distance(Ran begin, Ran end, std::random_access_iterator_tag dummy){
  std::cout << "RANDOM" << std::endl;
  return end - begin;
}

template <class I> inline
typename std::iterator_traits<I>::difference_type my_distance_wrapper(I begin, I end){
  typedef typename std::iterator_traits<I>::iterator_category cat;
  return my_distance(begin, end, cat());
}

int main() {
  std::vector<int> ve;
  std::list<int> li;
  for(int i = 0; i < 3; i++){
    ve.push_back(i);
    li.push_back(i);
  }
  std::cout << my_distance_wrapper(ve.begin(), ve.end()) << std::endl;
  std::cout << my_distance_wrapper(li.begin(), li.end()) << std::endl;
  return 0;
}

我认为我可以通过使用类似于std::is_random_acces_iterator<In>::value的{​​{1}}之类的功能来实现此目的 我检查了type_traits但是找不到任何东西来检查某些东西是否是特定的迭代器。 我如何检查是否有random_acces_iterator?我知道我可以在一个函数中执行此操作:

std::is_pod<In>::value

所以我的问题基本上是如何在模板中执行函数f的?不能不使用enable_if是不可能的,这是我的任务的要求。我是否正确相信SFINAE会正确地丢弃其他函数,如果我可以将函数f放入该模板中?

1 个答案:

答案 0 :(得分:5)

您可以使用类型特征std::is_same<>,如下所示

template<typename iterator>
using is_random_access_iterator = 
  std::is_same<typename std::iterator_traits<iterator>::iterator_category,
               std::random_access_iterator_tag>;

然后

template<typename iterator>
std::enable_if_t<is_random_access_iterator<iterator>::value,
                 typename std::iterator_traits<iterator>::difference_type>
my_distance(iterator a, iterator b) { return a-b; }

template<typename iterator>
std::enable_if_t<!is_random_access_iterator<iterator>::value,
                 typename std::iterator_traits<iterator>::difference_type>
my_distance(iterator a, iterator b) { /* ... */ }

这里,std::enable_if_t<,>是一个辅助别名(从C ++ 14开始)定义为

template<bool condition, typename type = void>
using enable_if_t = typename enable_if<condition,type>::type;

实际上,您也可以将功能f()声明为constexpr并直接在enable_if内使用,即std::enable_if<f<It>(), ...>