多继承的实例介绍

文章介绍了C++中多继承时遇到的同名覆盖问题,如何通过作用域解析符解决,以及二义性问题的处理,通过虚基类实现成员共享。此外,还讨论了赋值兼容性的规则,包括子类对象赋值给父类、初始化父类引用和指针的情况。
摘要由CSDN通过智能技术生成

一、多继承同名覆盖

子类中的成员与父类中的成员同名问题,通过作用域分辨符(::)进行限定类的访问,从而实现对不同类中的同名成员各自赋值。

#include<iostream>
using namespace std;
class A{//父类A 
	public:
		int val;
		void fun(){
			cout<<"Member of A,val值为:"<<val<<endl; 
		}
};
class B{//父类B:两父类的成员名相同 
	public:
		int val;
		void fun(){
			cout<<"Member of B,val值为:"<<val<<endl; 
		}
};
class C:public A,public B{//继承父类AB且成员名与父类的成员名相同 
	public:
		int val;
		void fun(){
			cout<<"Member of C,val值为:"<<val<<endl;
		}
};
int main()
{
	C c;
	c.val=1;//访问C类中的val变量 
	c.A::val=2;//通过作用域分辨符(::)限定访问成员的类 
	c.B::val=3;//限定访问B类中的val变量 
	c.fun();
	c.A::fun();//限定访问A类中的fun方法 
	c.B::fun(); 
 }

运行结果
在这里插入图片描述

二、多层多继承二义性问题

多层继承:
在这里插入图片描述

#include<iostream>
using namespace std;
class A{
    public:
	   int val;
	   void fun(){
		  cout<<"Member of A,val="<<val<<endl;
	    }
};
class B:public A{//B继承A 
	public:
		int val1;
};
class C:public A{//C继承A
	public:
		int val2;
};
class D:public B,public C//D继承B、C: 类D拥有两份在不同类下的同名拷贝
//二义性问题:对同一成员不同数据的多份拷贝,同一成员拥有不同数据 
{
	public:
		int val3;
};
int main()
{
	D d;
	d.B::val=3;//D中有两份不同的val与fun数据 
	d.C::val=5;
	d.B::fun();
	d.C::fun();
}

运行结果
在这里插入图片描述
出现二义性问题,即A类的同名成员有两份数据保存在内存中,为解决该问题,我们引入虚基类来标识其共同基(共同父类)来实现。

改进:将共同基(共同父类)定义为虚基类

#include<iostream>
using namespace std;
class A{
	public:
		int val;
		void fun(){
			cout<<"Member of A,val="<<val<<endl;
		}
};
class B:virtual public A{//B继承A 
	public:
		int val1;
};
class C:virtual public A{//C也继承A:将两者共同父类设为虚基类,可实现同名成员在内存中只有一份拷贝 
	public:
		int val2;
};
class D:public B,public C{
	public:
		int val3;
};
int main()
{
	D d;
	d.B::val=3;
	d.C::val=5;//新赋的值会覆盖原来赋的值,val变量在内存中只会有一个值 
	d.B::fun();
	d.C::fun();
	d.fun(); 
 } 

运行结果
在这里插入图片描述
通过虚基类实现同名成员的同一赋值。

三、赋值兼容性规则问题

赋值的单向性规则:
1.子类对象赋值给父类
2.子类对象初始化父类引用
3.子类对象地址赋给父类指针,且此时父类指针只能调用父类中的方法。

#include<iostream>
using namespace std;
class A{
	protected: 
	    int x;
	public:
		A(int x1=0):x(x1){
		}
		void print(){
			cout<<"A类中x="<<x<<endl;
		}
};
class B:public A{
	int y;
	public:
		B(int x1=0,int y1=0):A(x1){//子类B初始化父类A的引用
			y=y1;
		}
		void print(){
			cout<<"B类中x="<<x<<"\ty="<<y<<endl;
		}
};
class C:public A{
	int z;
	public:
		C(int x1=0,int z1=0) :A(x1){//子类C初始化父类A的引用 
			z=z1;
		}
		void print(){
			cout<<"C类中x="<<x<<"\tz="<<z<<endl;
		}
};
int main()
{
	A a(2);
	B b(3,4);//子类b对象将3赋值给A类中的x成员 
	C c(5,6);//子类c对象将5赋值给A类中的x成员 
	A *p[3]={&a,&b,&c};//定义父类A的指针,分别装B,C对象的地址 
	for(int i=0;i<3;i++)
	p[i]->print();//只能调用父类中的方法 
	b.print();
	c.print();
 }

运行结果
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值