【发布时间】:2021-03-26 22:32:28
【问题描述】:
struct B {
void f();
private:
B(int, int = 0);
};
struct D : B { using B::B; };
void B::f() {
auto a = D{0};
auto b = D(0);
auto c = D(0, 0);
D x{0};
D y(0);
D z(0, 0);
}
GCC 接受(从 7.1 开始;以前拒绝所有)。
Clang 接受 b 和 xyz,但拒绝 a 和 c。
MSVC 在 C++14 模式下拒绝所有,但在 C++17 模式下接受所有。
哪些编译器是正确的? C++14 和 C++17 之间的规则是否改变了 - 如果是这样,为什么在 C++14 中 B::f 不允许访问其自己的构造函数(通过 D 命名)?为什么 Clang 只接受 b 的(函数样式)强制转换,而不接受 a 的列表初始化或 c 的构造函数调用?
(使B 成为friend 的D 使Clang 接受,但旧版本(3.8 及更早版本)和C++14 模式下的MSVC 除外。它对gcc 没有影响。)
现在,我知道C++14 says:
如此声明的[作为继承构造函数]的构造函数与[基类]X中的相应构造函数具有相同的访问权限。
而在 C++17 中,规则得到了简化、澄清和moved to [namespace.udecl]:
命名构造函数的 using-declarator 不会创建同义词;相反,如果附加构造函数在用于构造相应基类的对象时可访问,则它们是可访问的,并且忽略 using 声明的可访问性。
所以,我认为也许“具有相同的访问权限”意味着继承的构造函数是 private 到 D 所以只有 D(及其朋友)可以访问,而不是 private 到 B (B::f 可以访问),B 中对应的构造函数也是如此。但在这种解释下,人们可以通过继承来颠覆访问检查并访问基类的私有构造函数。所以可以肯定的是,在 C++14 中,措辞的意图与在 C++17 中更清楚地表达的意图相同——但是为什么 MSVC 会根据语言版本改变其行为呢?
【问题讨论】:
标签: c++ c++17 language-lawyer crtp inheriting-constructors