时间:2021-05-20
在派生类中对基类成员访问应该是唯一的,但是在多继承时,可能会导致对基类某成员访问出现不一致的情况,这就是C++多继承中的二义性。
有两种继承的情况会产生多义性
一、如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的成员变量进行访问时,可能产生二义性,继承关系如下图所示:
#include <iostream>using namespace std; class A{public: int a;}; class B1 : public A{public: int b1;}; class B2 : public A{public: int b2;}; class C : public B1, public B2{public: int c;};int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c = 300; c1.a = 500; //报错,不能确定成员变量a具体在那个类 cout << "end..." << endl; system("pause"); return 0;}解决方法:虚继承
注意:C++编译系统在实例化C类时,只会将虚基类A的构造函数调用一次,忽略虚基类的其他派生类(class B1,class B2)对虚继承的构造函数的调用,从而保证了虚基类的数据成员不会被多次初始化。
在虚基类A中有一个虚指针指向一个虚表,虚表中记录了虚基类与本类的地址偏移,通过这个地址偏移可以找到虚基类的成员变量a的地址
#include <iostream>using namespace std; class A{public: int a;}; class B1 : virtual public A{public: int b1;}; class B2 : virtual public A{public: int b2;}; class C : public B1, public B2{public: int c1;}; int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c1 = 300; c1.a = 500; //虚继承使得成员变量a只有一份拷贝,通过虚指针可以确定地址 cout << "end..." << endl; system("pause"); return 0;}二、一个派生类同时继承两个基类,这两个基类存在相同的成员函数
#include<iostream>using namespace std; class Base1 {public: void fun() { cout << "I am base-1 " << endl; };}; class Base2{public: void fun() { cout << "I am base-2 " << endl; };}; class A: public Base1, public Base2 {public: void print() { }}; int main() { A obj; obj.fun(); //调用的方法产生歧义 system("pause"); return 0;}解决方法:使用作用域运算符::来解决,明确指向的方法,例如obj.Base1::fun()
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
本文实例讲述了C++多重继承引发的重复调用问题与解决方法。分享给大家供大家参考,具体如下:前面简单介绍了一个C++多重继承功能示例,这里再来分析一个多重继承引发
本文实例讲述了C++实现的多重继承功能。分享给大家供大家参考,具体如下:多重继承1.多重继承即一个类继承了多个基类的属性。2.多重继承下派生类的构造函数必须同时
源码if(key==null||value==null)thrownewNullPointerException();二义性假定ConcurrentHashMa
这篇文章主要介绍了Spring自动装配的二义性实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下1.我们
前言菱形继承是多重继承中跑不掉的,Java拿掉了多重继承,辅之以接口。C++中虽然没有明确说明接口这种东西,但是只有纯虚函数的类可以看作Java中的接口。在多重