条件constexpr函数

conditional constexpr functions

本文关键字:函数 constexpr 条件      更新时间:2023-10-16

这里有一个函数可能是constexpr的例子。通常情况下,只有在上下文允许的情况下,才添加constexpr并使用常量求值。然而,尽管没有在constexpr上下文中使用它,但以下代码仍然存在问题:

template <typename T>
struct Wrapper
{
friend constexpr bool operator==(const Wrapper& crLhs, const Wrapper& crRhs) noexcept
{
return crLhs.m_t == crRhs.m_t;
}
T m_t = {};
};

使用Visual Studio 2017 15.9.20,例如为std::string实例化时,会出现"错误C3615:consexpr函数"运算符=="无法生成常量表达式"。信息是正确的,但我不是在constexpr上下文中实例化它。

void f()
{
bool b;
Wrapper<int>  a;
b = a == a; //ok
Wrapper<std::string> c;
b = c == c;  //C3615, but not using constexpr context
}

我可以通过使用成员模板或删除constexpr来应用一个变通方法,但这里有没有两全其美的妙招(即适用时的constexpr(?

您的代码很好。这是旧版本的MSVC中的一个错误

此错误已在MSVC 19.22版本中修复,编译时没有出现错误。在这里,我们可以看到两个编译器版本的并排:https://godbolt.org/z/79kXFm

19.22之后的所有版本(包括19.22(都会编译它,但19.21及以下的版本错误地给出了错误C3615,尽管它们都设置为使用C++11。

GCC和Clang从未出现过这个错误

这个错误只出现在MSVC中,即使是非常旧的GCC和Clang版本也会编译代码而不会出错。

你该怎么办

如果可能的话,您应该转移到Visual Studio的较新版本。这是升级编译器的最简单选项,如果您转到新版本,编译器应该会得到错误修复和升级。如果这不是一个选项,我会用谷歌搜索不同的方法来升级编译器本身。这可能会有所帮助。