【发布时间】:2017-03-23 23:52:18
【问题描述】:
似乎 C++11 和 C++14 对待纯右值的 cv 限定不同。
C++11 坚持自 C++98 以来一直存在的“经典”方法:根据 3.10/4 “非类纯右值始终具有 cv 非限定类型”。
C++14 在 3.10/4 中包含类似的措辞,但它以注释的形式呈现:"[注意:类和数组纯右值可以有 cv 限定类型;其他纯右值总是有 cv 非限定类型. 见第 5 条。——尾注]"
在第 5 条中它说:
6 如果纯右值最初的类型为“cv T”,其中 T 是 cv 不合格的非类、非数组类型,则表达式的类型在任何进一步的分析。1
这个 5/6 条目是 C++14 中的新条目。它现在使用与引用类型结果相同的方法处理纯右值的 cv 限定(参见 5/5)。
这种变化的原因可能是什么? C++11 及之前的版本拒绝非类纯右值拥有任何 cv 资格的权利。 C++14 表示非类、非数组纯右值可以具有 cv 限定,但这些 cv 限定在在任何进一步分析之前被丢弃。
我的猜测是,有一些新的(针对 C++14 的)语言特性可以在适当的情况下(在上述调整发生之前)以某种方式“看到”纯右值的 cv 限定。它们存在吗?如果有,这些功能是什么?2
问题源于以下上下文:想象一个编译器在内部将 X 类的隐藏参数 this 实现为 X *const 类型的变量。由于编译器需要将this 公开为纯右值,因此const 不应该在C++11(或之前)中导致任何问题,其中标量纯右值永远不是cv 限定的。但是 C++14 呢?如果同一个编译器将this 暴露为X *const 类型的纯右值,会不会导致问题?
1 C++14 中的 5/6 和 3.10/4 中的注解似乎有矛盾,但注解无论如何也不是规范的。我正在使用文本的草稿版本。
2 我最初的猜测是decltype。而且我什至以为我在尝试的时候找到了答案
std::cout << std::is_same<decltype((const int) 0), const int>::value << std::endl;
在 GCC 中,输出 1。但是,看到 Clang 和 VC++ 输出 0(并且 decltype 的规范似乎不支持这种行为)我倾向于认为这只是 GCC 中的一个错误(从 6.1 开始)
【问题讨论】:
-
@Neil Butterworth:这是一个关于概念行为的问题。我问这个问题的全部原因是我无法找到任何可以利用 C++11 和 C++14 之间的上述差异的代码示例。如果差异无法通过代码来演示,那么我想知道是什么促使标准文本发生了变化。如果差异可以通过代码来演示,那么很高兴看到该代码。
-
@Thomas 这绝对是语言律师领域。
-
@Neil Butterworth:太好了。那么,新措辞“更准确或更细致”在哪些方面?这基本上就是我的问题所在。
-
这是核心问题 1261,在公共问题列表中的讨论为零,但我认为这只是措辞清理。例如,给定
const int f();,f()是一个prvalue,根据[expr.call],它的类型是“静态选择函数的返回类型”,即const int,某种措辞是需要让const消失。 -
嗯,是的,GCC 曾经遇到过
S*&& p = this;之类的代码问题。现在他们没有了。
标签: c++ c++11 c++14 language-lawyer