用std :: enable_if重载函数以避免模板替换错误

时间:2019-04-17 14:18:05

标签: c++ c++17 enable-if std-variant

我想编写两个模板函数,以便一个捕获特定情况,另一个捕获与第一个情况不匹配的所有其他情况。 我正在尝试使用std :: enable_if来捕获特定情况,但是编译器仍然因模棱两可的匹配而失败。如何编写这些重载函数,以便编译器解决歧义? (我正在使用g ++)

我尝试编写以下代码(这是一个重现问题的简化示例):

struct resource1_t{
};

struct resource2_t{
};

template <typename R, typename V>
struct unit_t{
  typedef R resource_t;
  typedef V value_t;
  unit_t(value_t const& value):v(){}
  value_t v;
  value_t calcValue(resource_t const& r)const{return v;}
};

// Specific case (U::resource_t == R)
template <typename U, typename R, typename=std::enable_if_t<std::is_same_v<typename U::resource_t,R>>>
      typename U::value_t callCalcValue(U const& u, R const& r){
        return u.calcValue(r);
      }

 // General case (U::resource_t != R)
 template <typename U, typename R>
      typename U::value_t callCalcValue(U const& u, R const& r){
        // Fail immediately!
        assert(!"Unit resource does not match");
        return U::value_t();
      }

int main()
{
    // Create an array of unit variants
   typedef unit_t<resource1_t,int> U1;
   typedef unit_t<resource2_t,float> U2;
   std::vector<std::variant<U1,U2>> units;
   units.emplace_back(U1(1));
   units.emplace_back(U2(1.0f));

   // Create a parallel array of resources
   std::vector<std::variant<resource1_t,resource2_t>> resources;
   resources.emplace_back(resource1_t());
   resources.emplace_back(resource2_t());

   // Call calcValue for each unit on the parallel resource
   for(int i(0); i<units.size(); ++i){
       std::visit([&](auto&& unit){
           std::visit([&](auto&& resource){
             // Fails to compile with substitution failure...
             //std::cout << unit.calcValue(resource) << "\n";

             // Results in ambiguous call compile error...
             std::cout << callCalcValue(unit,resource) << "\n";
           },resources[i]);
       },units[i]);
   }
}

我希望编译器将std::is_same_v<U::resource_t,R>的所有情况与特定情况匹配,并将所有其他组合与一般情况进行匹配,相反,编译器无法说出该函数是模棱两可的。 我还尝试了! std::is_same作为第二个定义,并且编译器因error: redefinition of ... callCalcValue()...

而失败

2 个答案:

答案 0 :(得分:2)

这是一个简化的例子:

template <typename T, typename R, typename=std::enable_if_t<std::is_same_v<R, int>>
void foo(T, R); // #1

template <typename T, typename R>
void foo(T, R); // #2

foo(some_t{}, some_r{});

特殊的sfinae约束并不重要,因此我选择了一个简单的约束。现在,如果满足约束条件(在这种情况下为is_same_v<R, int>),则替换为#1将会失败,我们只剩下一个候选对象:#2

但是,如果替换成功,那么我们就有两个候选人。它们具有同等的可行性,没有任何区别。因此,通话不明确!


我们需要一种其他方法来区分它们。一种方法是将取反的约束添加到另一个重载中(请注意,您需要在此处更改SFINAE表格才能使之工作):

template <typename T, typename R, std::enable_if_t<std::is_same_v<R, int>, int> = 0>
void foo(T, R); // #1

template <typename T, typename R, std::enable_if_t<!std::is_same_v<R, int>, int> = 0>
void foo(T, R); // #2

这可以确保两者中的一个完全可行。

另一种方法是将特征转发到单独的重载集:

template <typename T, typename R>
void foo_impl(T, R, std::true_type); // #1

template <typename T, typename R>
void foo_impl(T, R, std::false_type); // #2

template <typename T, typename R>
void foo(T t, R r) {
    return foo_impl(t, r, std::is_same<R, int>{}); // NB: not _v
}

另一种方法是只编写一个重载并在内部使用if constexpr

template <typename T, typename R>
void foo(T, R) {
    if constexpr (std::is_same_v<R, int>) {
        // #1
    } else {
        // #2
    }
}

(在C ++ 20中)未来的方式是使用概念:

template <typename T, typename R>
    requires std::is_same_v<R, int>
void foo(T, R); // #1

template <typename T, typename R>
void foo(T, R); // #2

如果可行,这将使#1#2更具约束力,因此它是首选。

答案 1 :(得分:1)

我想知道简单的重载是否也可以工作:

// General case (U::resource_t != R)
template <typename U, typename R>
     typename U::value_t callCalcValue(U const& u, R const& r){
       // Fail immediately!
       assert(!"Unit resource does not match");
       return U::value_t();
     }

//Specific case (U::resource_t == R) overloads the general case
template <typename U>
      typename U::value_t callCalcValue(U const& u,  typename U::resource_t const& r){
        return u.calcValue(r);
      }