How does virtual inheritance solve the quot;diamondquot; (multiple inheritance) ambiguity?(虚拟继承如何解决“钻石问题?(多重继承)歧义?)
问题描述
class A { public: void eat(){ cout<<"A";} };
class B: virtual public A { public: void eat(){ cout<<"B";} };
class C: virtual public A { public: void eat(){ cout<<"C";} };
class D: public B,C { public: void eat(){ cout<<"D";} };
int main(){
A *a = new D();
a->eat();
}
我理解钻石问题,上面的代码没有这个问题.
I understand the diamond problem, and above piece of code does not have that problem.
虚拟继承究竟是如何解决问题的?
How exactly does virtual inheritance solve the problem?
我的理解:当我说 A *a = new D(); 时,编译器想知道是否可以将 D 类型的对象分配给 A 类型的指针,但它有两条路径可以走,但不能自己决定.
What I understand:
When I say A *a = new D();, the compiler wants to know if an object of type D can be assigned to a pointer of type A, but it has two paths that it can follow, but cannot decide by itself.
那么,虚拟继承是如何解决这个问题的(帮助编译器做出决定)?
So, how does virtual inheritance resolve the issue (help compiler take the decision)?
推荐答案
您想要:(可通过虚拟继承实现)
You want: (Achievable with virtual inheritance)
A
/
B C
/
D
而不是:(没有虚拟继承会发生什么)
And not: (What happens without virtual inheritance)
A A
| |
B C
/
D
虚拟继承意味着只有 1 个 A 基类的实例,而不是 2 个.
Virtual inheritance means that there will be only 1 instance of the base A class not 2.
您的类型 D 将有 2 个 vtable 指针(您可以在第一个图中看到它们),一个用于 B,一个用于 C他们实际上继承了A.D 的对象大小增加了,因为它现在存储了 2 个指针;但是现在只有一个A.
Your type D would have 2 vtable pointers (you can see them in the first diagram), one for B and one for C who virtually inherit A. D's object size is increased because it stores 2 pointers now; however there is only one A now.
所以 B::A 和 C::A 是相同的,所以不会有来自 D 的歧义调用.如果你不使用虚拟继承,你有上面的第二个图.任何对 A 成员的调用都会变得模棱两可,您需要指定要采用的路径.
So B::A and C::A are the same and so there can be no ambiguous calls from D. If you don't use virtual inheritance you have the second diagram above. And any call to a member of A then becomes ambiguous and you need to specify which path you want to take.
维基百科有另一个很好的纲要和例子
这篇关于虚拟继承如何解决“钻石"问题?(多重继承)歧义?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:虚拟继承如何解决“钻石"问题?(多重继承)歧
基础教程推荐
- 如果我为无符号变量分配负值会发生什么? 2022-01-01
- 为什么 typeid.name() 使用 GCC 返回奇怪的字符以及如 2022-09-16
- GDB 显示调用堆栈上函数地址的当前编译二进制文 2022-09-05
- 非静态 const 成员,不能使用默认赋值运算符 2022-10-09
- 初始化列表*参数*评估顺序 2021-01-01
- 为什么派生模板类不能访问基模板类的标识符? 2021-01-01
- CString 到 char* 2021-01-01
- 为什么 RegOpenKeyEx() 在 Vista 64 位上返回错误代码 2021-01-01
- 我应该对 C++ 中的成员变量和函数参数使用相同的名称吗? 2021-01-01
- 通过引用传递 C++ 迭代器有什么问题? 2022-01-01
