2

给定以下代码:

template<typename T>
constexpr remove_reference_t<decltype(T{}.x, bool{})> has_x() {return true;}
template<typename T, class... U>
constexpr bool has_x(U...) {return false;}

class A { public: int x; };

int main()
{
    vector<int> vec;
    A my_a{};

    std::cout << has_x<decltype(my_a)>() << endl << has_x<decltype(vec)>() << endl;

    if constexpr(has_x<decltype(vec)>())
    {
        cout << vec.x << endl;
    }
    else
    {
        cout << size(vec) << endl;
    }
}

只有当我注释掉cout << vec.x << endl. 这显然不会编译,但我的理解if constexpr是:

如果值为true,则丢弃 statement-false(如果存在),否则丢弃 statement-true

因此,我认为应该丢弃“statement-true”,但似乎并非如此。如果我在“statement-true”中放置一个在任何一种情况下都有效的陈述,它就会起作用。但是我得到一个可能无效的声明:

错误:class std::vector<int>没有名为的成员x

我在这里做错了吗?

Live Example

4

1 回答 1

3

使用 constexpr if,被丢弃的语句的主体在语法上仍然必须是正确的,即使它没有被编译。编译器在编译时知道

vec.x

是不正确的,所以你得到一个错误。如果您重构代码以使用类似的模板

template<typename T>
void foo(T& vec)
{
    if constexpr(has_x<T>())
    {
        cout << vec.x << endl;
    }
    else
    {
        cout << size(vec) << endl;
    }
}

int main()
{
    vector<int> vec;
    A my_a{};

    std::cout << has_x<decltype(my_a)>() << endl << has_x<decltype(vec)>() << endl;
    foo(vec);

}

然后

vec.x

在语法上是正确的,它不知道是什么vec,但它不是格式错误的代码,所以它通过了。然后,一旦模板被实例化,if 语句的条件被评估并被vec.x丢弃,因此没有编译器错误。


通常,您只在模板上下文中使用 constexpr。您不必这样做,但如果您不必这样做,则必须确保主体将在编译时编译,即使它会被丢弃,就像普通的旧 if 语句一样。


即使在模板内部,您仍然必须小心。如果 constexpr if 的主体不依赖于模板参数,那么它将在模板被实例化之前被评估。使用

template <typename T>
void f() 
{
     if constexpr (std::is_integer_v<T>)
         // ...
     else
       static_assert(false, "T must be an integer type"); 
}

代码不会编译,因为static_assert(false, "T must be an integer type")在解析模板时触发。您必须使条件取决于模板类型,以便在实例化时对其进行评估,例如

template<class T> struct always_false : std::false_type {};

template <typename T>
void f() 
{
     if constexpr (std::is_integer_v<T>)
         // ...
     else
       static_assert(always_false<T>, "T must be an integer type"); 
}
于 2019-03-18T20:41:41.057 回答