Ad-hoc constraint requires requires requires requires?(临时约束是否需要?)
问题描述
假设我正在编写一个类模板,其某些成员受类型模板参数静态常量数据成员的存在和值的约束:
template<class T>
struct A {
constexpr bool operator()() requires T::value { return T::value; }
constexpr bool operator()() { return false; }
};
#include <type_traits>
static_assert(A<std::true_type>()());
static_assert(!A<std::false_type>()());
static_assert(!A<void>()());
msvc和GCC接受这一点,但clang拒绝,除非我replacerequires T::value
requires requires { requires T::value; }
。这是clang中的一个错误,还是其他编译器松懈;这是C++所要求的情况吗?《标准》怎么说?
相关(至少是⅔):Why do we require requires requires?
推荐答案
这是一个clang错误(已提交#49513)。
看起来clang正在将void
替换为T::value
,并确定因为这是无效的表达式,所以约束无效。但[temp.constr.atomic]中的规则是:
要确定是否满足原子约束,首先将参数映射和模板实参替换到其表达式中。如果替换导致无效的类型或表达式,则不满足约束。
在这种情况下,替换会导致无效的类型或表达式,因此结果应该是不满足约束。
请注意,此重载:
constexpr bool operator()() requires T::value { return T::value; }
仅当T::value
是有效的表达式并且计算为true
时才有效。这就相当于:
constexpr bool operator()() requires T::value { return true; }
在这种情况下没有问题,因为在另一种情况下返回的是false
,所以没有理由区分T::value
存在,但false
和T::value
不存在。
但认为无论如何都值得澄清。
这篇关于临时约束是否需要?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:临时约束是否需要?
基础教程推荐
- 设计字符串本地化的最佳方法 2022-01-01
- C++,'if' 表达式中的变量声明 2021-01-01
- C++ 标准:取消引用 NULL 指针以获取引用? 2021-01-01
- 运算符重载的基本规则和习语是什么? 2022-10-31
- C++ 程序在执行 std::string 分配时总是崩溃 2022-01-01
- 如何在 C++ 中处理或避免堆栈溢出 2022-01-01
- 什么是T&&(双与号)在 C++11 中是什么意思? 2022-11-04
- 调用std::Package_TASK::Get_Future()时可能出现争用情况 2022-12-17
- 您如何将 CreateThread 用于属于类成员的函数? 2021-01-01
- 如何定义双括号/双迭代器运算符,类似于向量的向量? 2022-01-01