【发布时间】:2017-06-19 09:08:49
【问题描述】:
class A { public: int a; };
class B : public virtual A { public: using A::a; };
class C : public virtual A { public: using A::a; };
class D : public C, public B { };
class W { public: int w; };
class X : public virtual W { public: using W::w; };
class Y : public virtual W { };
class Z : public Y, public X { };
int main(){
D d;
d.a = 0; // Error
Z z;
z.w = 0; // Correct
return 0;
}
第一组类声明(A、B、C 和 D)和第二组(W、X、Y 和 Z)的构建方式类似,除了class C 有一个 using 声明 (using A::a) 而 class Y 没有。
当尝试在 d.a = 0 中访问成员 a 时,我在 Clang (error: member 'a' found in multiple base classes of different types) 和 GCC (error: request for member 'a' is ambiguous) 中遇到错误。我检查了这两个编译器的最新版本和旧版本,他们都同意。但是,在z.w = 0 中访问w 编译成功。
访问a时出现这种歧义的原因是什么?
据我所知,B 和 C 类中的访问声明都引用同一个基类成员。顺便说一句,如果我删除它们,测试会成功编译,因为 a 已经可以公开访问(public 访问说明符)。
提前致谢。
注意:以上代码是来自SolidSands' SuperTest 套件的稍微修改的测试。
【问题讨论】:
-
d.a可以在d.B::a和d.C::a之间进行选择(即使实际上是一样的)。 -
为什么不使用
d.A::a = 0;? -
我在cppreference 找不到任何适用于此处的
using文档(仅别名和与命名空间相关的用途)。 -
C++11 中成员访问规则发生了变化。据我所知,您的示例对 C++11 完全有效。 C++03 规则 我相信 使您的第一个代码格式错误,因为它不考虑使用透明的声明(查找不通过它们查找 C++03):它将找到“w " 在 B 和 C 中,没有一个比另一个更占优势。对于第二个测试用例,它在 W 和 X 中找到“w”,但 X 比 W 更占优势,因此 X 获胜。在 C++11 中,在两个测试用例中,它分别在 W 和 A 中找到“w”,然后就没有歧义了。
-
@jos 是的,所以我不知道,当时似乎在 C++03 中也有效。另一方面,它并没有说这两个 using 声明是合并的。它只是说两者都被认为来自同一个子对象(出于这两个检查的目的)。所以名称查找仍然会找到两个声明。也许这就是编译器抱怨的原因?
标签: c++ c++11 c++14 member c++03