enable_if有条件地包含成员函数

Ben*_*nes 6 c++ templates enable-if c++11

我有一个模板类,其类型是迭代器.我想根据模板参数的iterator_category启用/禁用特定成员函数.特别是,operator--如果模板参数是双向迭代器,我想启用.我的尝试是这样的:

    typename std::enable_if<
       std::is_base_of<std::bidirectional_iterator_tag,
                    MyTemplateParameter>::value,
    MyType&>::type
    operator --() {
    //do work
    return *this;
  }
Run Code Online (Sandbox Code Playgroud)

Clang告诉我(粗略地): error: no type named 'type' in 'std::__1::enable_if<false, MyTemplateParameter>'; 'enable_if' cannot be used to disable this declaration

有没有办法完成我正在尝试的东西?

以下是某些上下文中的示例:

    #include <iterator>
    #include <type_traits>

    template <typename TagType> 
    class test {
      public:
      typename std::enable_if<
         std::is_base_of<std::bidirectional_iterator_tag,
                        TagType>::value,
        test>::type
      operator --() {
         return *this;
      }

    };

    int main(){

      test<std::random_access_iterator_tag> t1;
      test<std::forward_iterator_tag> t2;

    /*
    breakTemps.cpp:13:2: error: no type named 'type' in 'std::__1::enable_if<false,     test<std::__1::forward_iterator_tag> >'; 'enable_if' cannot be used to disable this declaration
            std::is_base_of<std::bidirectional_iterator_tag,
            ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    breakTemps.cpp:25:35: note: in instantiation of template class 'test<std::__1::forward_iterator_tag>' requested here
       test<std::forward_iterator_tag> t2;
                                       ^
    */

}
Run Code Online (Sandbox Code Playgroud)

小智 11

std::enable_if 需要依赖于成员模板本身的参数.

template <typename TagType>
class foo
{
public:
    template <typename U = TagType>
      typename std::enable_if<
         std::is_base_of<std::bidirectional_iterator_tag,
                        U>::value,
        foo>::type
      operator --() {
         return *this;
      }
};
Run Code Online (Sandbox Code Playgroud)

SFINAE将按预期工作.

int main() {
  foo<std::random_access_iterator_tag> f;
  foo<std::forward_iterator_tag> f2;
  --f; // fine
  --f2;
}

main.cpp:24:3: error: no match for 'operator--' (operand type is 'foo<std::forward_iterator_tag>')

--f2;
Run Code Online (Sandbox Code Playgroud)

  • @BenJones 没有“模板”函数类型替换,就没有失败的替换。该技术基于模板类型替换失败的副作用:如果替换了 Vector=int,则 template&lt;class Vector&gt; typename Vector::iterator abs( Vector const&amp; )` 将导致构建失败。SFINAE 表示,这种错误应该不考虑过载。实际上,C++ 用户正在侵入一个并非其目的的系统来执行其他操作,这就是 SFINAE 有点尴尬的原因。传入的概念...就是为了做到这一点! (2认同)