C++:多继承中的二义性问题

    在派生类中对基类成员的访问应该是唯一的.但是,在多继承情况下,可能造成对基类中某个成员的访问出现了不一致的情况,这时就称对基类成员的访问产生了二义性.

原因之一:

   派生类在访问基类成员函数时,由于基类存在同名的成员函数,导致无法确定访问的是哪个基类的成员函数,因此出现了二义性错误。

例如:

	#include<iostream>
	using namespace std;

	class Base1
	{
	public:
		void fun(){cout<<"base1 "<<endl;};
	};

	class Base2
	{
	public:
		void fun(){cout<<"base2 "<<endl;};
	};

	class Derived:public Base1,public Base2{};

	int main()
	{
		Derived obj;
		obj.fun();   //产生歧义
		return 0;
	}

编译时,会提示:

当派生类Derived访问fun()函数时,无法确定访问的是Base1的还是Base2的,将出现二义性错误。

解决办法:

  1,使用作用域运算符指明访问的是base1的还是base2的fun函数。

         例如:

                 obj.Base1::fun();   //指明访问base1的fun函数

 

  2,在类中定义同名成员,使内层函数覆盖外层的函数。

         例如:

 

classDerived:public Base1,public Base2

{

public:

voidfun(){cout<<"使用的是内层的fun函数"<<endl;}

};

运行:

原因之二

     当一个派生类从多个基类派生时,而这些基类又有一个共同的基类,当对这个共同的基类中说明的成员进行访问时,可能出现二义性问题。

例如:

    Base为Derived11和Derived12的基类,而Derived2又继承Derived11和Derived12,当Derived2访问Base的data时,会出现二义性错误,代码如下:

class Base
{
public:
	int data;
};

class Derived11:public Base{};
class Derived12:public Base{};

class Derived2:public Derived11,public Derived12 {};

int main()
{
	Derived2 obj;
	obj.data=1;   //产生二义性
	return 0;
}

解决方法:

        1,作用域运算符

       Obj.Derived11::a,指明访问哪一个基类的data.但是由于派生类的直接基类有一个共同的基类,所以
obj.Base::a是错误的。

2,使用虚基类

产生二义性的最主要的原因就是base在派生类Derived2中产生了2个对象,从而导致了对基类Base的成员data访问的不一致性。要解决这个问题,只需使这个公共基类Base在派生类中只产生一个子对象即可。

 

   虽然在形式上,Derived2继承Derived11,Derived12,但是在存储结构上,对data的访问是指向Base的。

 

  总结:最近的学习和生活让我感觉到,生活中的很多事情都是千疮百孔的,但是我们有很多方法可以解决出现的问题。

   

  

   



时间: 2024-11-02 14:24:49

C++:多继承中的二义性问题的相关文章

Java语言入门教程(十三):Java语言中继承中的构造方法问题

教程(十一)中,了解了Java语言中继承的基本概念.Java中类与类的继承 ,是单继承,主要目的是复用.子类对象可以复用父类中权限允许的属性和方法 ,所以子类的构造方法和父类的构造方法之间,有一定的调用关系,本文中将进 行详细介绍. 首先,需要记住一个事实:子类的任何一个构造方法,都将先调用父类某个 构造方法.如子类Trainer中的构造方法: public Trainer() { } 虽然这个构造方法的方法体中什么代码也没有写,但是也调用了父类 Employee的构造方法,默认调用的是Empl

java 继承-一个java继承中碰到的菜鸟问题

问题描述 一个java继承中碰到的菜鸟问题 public class Parent { public String a; public String b; public Parent getA(){ System.out.println("PPPPPPPPP"); return this; } public void Test(){ System.out.println(getA().a); System.out.println(getA().b); } } public class

c++ 容器与继承-c++继承中拷备构造函数调用的疑问?

问题描述 c++继承中拷备构造函数调用的疑问? class Base { public: Base() { cout << "Base()" << endl; } Base(const Base&) { cout << "Base(const Base&)" << endl; } ~Base() { cout << "~Base()" << endl; } }

继承中子类的属性行为

问题描述 继承中子类的属性行为 继承 子类中是实际存在父类的属性和方法,还是只是能调用,实际没有 解决方案 继承好比说你对父类的代码进行了二次的封装, 类似于将父类的代码copy到自己的代码中,所以是实际存在父类的属性和方法! 解决方案二: java中子类继承了父类的哪些成员Java 中子类继承父类,执行顺序python中单继承和多继承中子类默认继承父类的哪个构造函数__init__

C++继承中的访问控制实例分析_C 语言

本文较为深入的探讨了C++继承中的访问控制,对深入掌握C++面向对象程序设计是非常必要的.具体内容如下: 通常来说,我们认为一个类有两种不同的用户:普通用户 和 类的实现者.其中,普通用户编写的代码使用类的对象,这部分代码只能访问类的公有(接口)成员:实现者则负责编写类的成员和友元的代码,成员和友元既能访问类的公有部分,也能访问类的私有部分.如果进一步考虑继承的话就会出现第三种用户,即派生类.派生类可以访问基类的公有(public)成员和受保护(protected)成员,但不能访问基类的私有(p

java继承中的构造方法实例解析_java

本文实例讲述了java继承中的构造方法.分享给大家供大家参考.具体如下: 继承中的构造方法: 1.子类的构造过程中必须调用其基类的构造方法. 2.子类可以在自己的构造方法中使用super(argument_list)调用基类的构造方法.   2.1.使用this(argument_list)调用本类的另外构造方法.   2.2.如果调用super,必须写在子类构造方法的第一行. 3.如果子类的构造方法中没有显示的调用基类的构造方法,则系统默认调用基类的无参数构造方法. 4.如果子类构造方法中既没

php构造方法中析构方法在继承中的表现_php技巧

本文为大家分享了php构造方法之析构方法在继承中的表现,供大家参考,具体内容如下 子类中没有定义构造方法时,会自动调用父类的构造方法.因此实例化子类时,需按照父类的构造方法的形式进行. 修改为: 子类定义了自己的构造方法,则不会自动调用父类的构造方法,但可以手动调用:parent::__construct(); 但通常,在子类,很多时候,在构造方法中,都应该(需要)去调用父类的构造方法以节省代码,增加可读性: 子类中没有定义析构方法时,会自动调用父类的析构方法.子类定义了自己的析构方法,则不会自

java继承中的super调用

问题描述 java继承中的super调用 最近看到了下面一段代码,都是覆盖父类的方法,为什么onCreate方法就需要调用super.onCreate(..),而onActivityResult(..)方法就不需要调用super.onActivityResult(..)呢?求哪位解释一下 thanks! public class MainActivity extends Activity { ....... @Override protected void onCreate(Bundle sav

js继承中的问题?toString()方法

问题描述 js继承中的问题?toString()方法 js中的原型的toString()方法,怎么理解呢?后面的call又是什么意思?谢谢大神们了 解决方案 由JS中toString()方法所想到的JS中的 toString 和 valueOf 方法继承中的构造方法,object类,toString类,equals方法