gpt4 book ai didi

c++ - 为什么 SFINAE (enable_if) 从类定义内部工作而不是从外部工作

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:18:57 26 4
gpt4 key购买 nike

过去几个小时我一直在努力解决一个非常奇怪的问题(在用 SFINAE 解决了 5-6 个其他问题之后,因为我是新手)。基本上在下面的代码中,我想让 f() 为所有可能的模板实例化工作,但是 g() 仅在 N == 2 :

#include <type_traits>
#include <iostream>

template<typename T, int N>
class A
{
public:
void f(void);
void g(void);
};

template<typename T, int N>
inline void A<T, N>::f()
{
std::cout << "f()\n";
}

template<typename T, int N, typename std::enable_if<N == 2, void>::type* = nullptr>
inline void A<T, N>::g()
{
std::cout << "g()\n";
}

int main(int argc, char *argv[])
{
A<float, 2> obj;
obj.f();
obj.g();

return 0;
}

当我尝试编译它时,我收到一个关于有 3 个而不是两个模板参数的错误。然后,经过一些试验,我决定将 g() 的定义移动到 A 本身的定义中,如下所示:

#include <type_traits>
#include <iostream>

template<typename T, int N>
class A
{
public:
void f(void);

template<typename t = T, int n = N, typename std::enable_if<N == 2, void>::type* = nullptr>
void g()
{
std::cout << "g()\n";
}
};

template<typename T, int N>
inline void A<T, N>::f()
{
std::cout << "f()\n";
}

int main(int argc, char *argv[])
{
A<float, 2> obj;
obj.f();
obj.g();

return 0;
}

现在,奇迹般地一切正常。但我的问题是为什么?编译器没有看到在类定义中我试图内联一个也依赖于 3 个模板参数的成员函数吗?或者让我们反过来问:如果它在 A 的定义内有效,为什么它在外部不起作用?哪里不一样?不是还有 3 个参数,这比类 A 需要的模板参数多 +1 吗?

另外,为什么它只在我将第三个参数设为非类型参数而不是类型参数时才起作用?请注意,我实际上创建了一个由 enable_if 返回的类型的指针,并为其分配了一个默认值 nullptr,但我发现我不能像我在这里看到的其他 SO 论坛帖子那样将它作为类型参数留在那里。

非常感谢,谢谢!!!

最佳答案

那是因为模板化类中的模板化函数有两组 模板参数集,而不是一组。因此,“正确”的形式是:

template<typename T, int N>
class A
{
public:
void f(void);

template<typename std::enable_if<N == 2, void>::type* = nullptr>
void g(void);
};

template<typename T, int N> // Class template.
template<typename std::enable_if<N == 2, void>::type* /* = nullptr */> // Function template.
inline void A<T, N>::g()
{
std::cout << "g()\n";
}

查看实际效果 here .

[请注意,这不是实际上正确的,原因在此答案的底部进行了解释。如果 N != 2 就会坏掉.]

如果您愿意,请继续阅读以获取解释。


还在我身边吗?好的。让我们检查每种情况,好吗?

  1. 定义 A<T, N>::g()外面A :

    template<typename T, int N>
    class A
    {
    public:
    void f(void);
    void g(void);
    };

    template<typename T, int N, typename std::enable_if<N == 2, void>::type* = nullptr>
    inline void A<T, N>::g()
    {
    std::cout << "g()\n";
    }

    在这种情况下,A<T, N>::g()的模板声明与 A 不匹配的模板声明。因此,编译器会发出错误。此外,g()本身不是模板化的,所以模板不能在不改变的情况下拆分为类模板和函数模板 A的定义。

    template<typename T, int N>
    class A
    {
    public:
    void f(void);

    // Here...
    template<typename std::enable_if<N == 2, void>::type* = nullptr>
    void g(void);
    };

    // And here.
    template<typename T, int N> // Class template.
    template<typename std::enable_if<N == 2, void>::type* /* = nullptr */> // Function template.
    inline void A<T, N>::g()
    {
    std::cout << "g()\n";
    }
  2. 定义 A<T, N>::g()里面A :

    template<typename T, int N>
    class A
    {
    public:
    void f(void);

    template<typename t = T, int n = N, typename std::enable_if<N == 2, void>::type* = nullptr>
    void g()
    {
    std::cout << "g()\n";
    }
    };

    在这种情况下,由于 g()是内联定义的,它隐式具有 A的模板参数,无需手动指定。因此,g()实际上是:

    // ...
    template<typename T, int N>
    template<typename t = T, int n = N, typename std::enable_if<N == 2, void>::type* = nullptr>
    void g()
    {
    std::cout << "g()\n";
    }
    // ...

在这两种情况下,对于 g()要拥有自己的模板参数,同时作为模板类的成员,函数模板参数必须与类模板参数分开。否则,函数的类模板将与类不匹配。


既然我们已经介绍过了,我应该指出 SFINAE 只关注立即模板参数。所以,对于 g()将 SFINAE 与 N 一起使用, N需要是它的模板参数;否则,如果您尝试调用,例如 A<float, 3>{}.g(),您将收到错误消息.如有必要,这可以通过中介来完成。

此外,您需要提供 g() 的版本当 N != 2 时可以调用.这是因为 SFINAE 仅在至少存在一个有效版本的函数时才适用;如果没有 g() 的版本可以调用,然后将发出错误并且不会执行 SFINAE。

template<typename T, int N>
class A
{
public:
void f(void);

// Note the use of "MyN".
template<int MyN = N, typename std::enable_if<MyN == 2, void>::type* = nullptr>
void g(void);

// Note the "fail condition" overload.
template<int MyN = N, typename std::enable_if<MyN != 2, void>::type* = nullptr>
void g(void);
};

template<typename T, int N>
template<int MyN /*= N*/, typename std::enable_if<MyN == 2, void>::type* /* = nullptr */>
inline void A<T, N>::g()
{
std::cout << "g()\n";
}

template<typename T, int N>
template<int MyN /*= N*/, typename std::enable_if<MyN != 2, void>::type* /* = nullptr */>
inline void A<T, N>::g()
{
std::cout << "()g\n";
}

如果这样做,我们可以通过让中介来完成繁重的工作来进一步简化事情。

template<typename T, int N>
class A
{
public:
void f(void);

template<bool B = (N == 2), typename std::enable_if<B, void>::type* = nullptr>
void g(void);

template<bool B = (N == 2), typename std::enable_if<!B, void>::type* = nullptr>
void g(void);
};

// ...

查看实际效果 here .

关于c++ - 为什么 SFINAE (enable_if) 从类定义内部工作而不是从外部工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41904099/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com