C++中菱形继承的解释与处理详解

 更新时间:2022年02月08日 09:01:14   作者:龙小  
菱形继承是多重继承中跑不掉的,Java拿掉了多重继承,辅之以接口。C++中虽然没有明确说明接口这种东西,但是只有纯虚函数的类可以看作Java中的接口,下面这篇文章主要给大家介绍了关于C++中菱形继承的解释与处理的相关资料,需要的朋友可以参考下

封装,继承,多态。这是C++语言的三大特性,而每次在谈到继承时我们不可避免的要谈到一个很重要的问题——菱形继承。

派生类继承父类,同时也会继承父类中的所有成员副本,但如果在继承时一个基类同时被两个子类继承,然后一个新类又分别由上面的两个子类派生出来。这样从某种程度来说就形成了C++中的菱形继承,也可以叫做钻石继承,具体的继承形式如下图所示:

在上面的类图说,Left和Right分别派生子Top,但是Bottom又分别继承了Left和Right。继承关系也可以画成下面的方式,这样就可以更好的理解设计中存在的问题。

该类图很明确的展示了类设计中的不足之处,在试图将指向Bottom对象的指针转换成指向Top的指针时,有两个Top对象可供选择,但是编译器却明显没有那么智能,从而导致了转换过程中的二义性;同理,Bottom对象也不能直接调用Top中定义的方法,如果要使用需要提供一个Top子对象,但是从类图可知存在两个Top对象。

上面的类对应的代码为:

class Top{
public:
    int _x;
public:
    Top(int x):_x(x){};
};
class Left:public Top{
public:
    int _y;
public:
    Left(int x,int y):Top(x),_y(y){}
};
class Right:public Top{
public:
    int _z;
public:
    Right(int x,int z):Top(x),_z(z){}
};
class Bottom:public Left,public Right{
public:
    int _w;
public:
    Bottom(int x,int y,int z,int w):Left(x,z),Right(y,z),_w(w){};
};

下面实现该类的测试程序,如下所示:

int main()
{
    Bottom bf(1,2,3,4);
    cout<<sizeof(bf)<<endl;
    return 0;
}

运行结果为:20,在打印基类中的成员时编译器也会报以下错误:

既然在上面的类的设计中存在问题,在实际编程时如何避免这个问题呢?

答案是:虚基类。

虚基类给在确实需要使用菱形继承的地方提供了一个很好的解决方法,通过子类共享一个基类对象避免基类对象的二义性问题。

上面的代码修改后代码如下:

using namespace std;
class Top{
public:
    int _x;
public:
    Top(int x):_x(x){};
    virtual ~Top(){};
};
class Left:virtual public Top{
public:
    int _y;
public:
    Left(int x,int y):Top(x),_y(y){}
};
class Right:virtual public Top{
public:
    int _z;
public:
    Right(int x,int z):Top(x),_z(z){}
};
class Bottom:public Left,public Right{
public:
    int _w;
public:
    Bottom(int x,int y,int z,int w):Top(x),Left(x,y),Right(x,z),_w(w){};
};

在main函数中继续测试上述类,则可以正常输出,代码如下:

int main()
{
    Bottom bf(1,2,3,4);
    cout<<bf._x<<","<<bf._y<<","<<bf._z<<","<<bf._w<<endl;
    return 0;
}

运行结果为:

从上面的示例可以看出,在使用多进程时如果不对类进行提前规划,将可能产生菱形继承这种场景,给实际的编程带来不便。因此在实际编码时,我建议尽量减少多继承的方式更多地使用嵌套类的方式。

总结

到此这篇关于C++中菱形继承的解释与处理的文章就介绍到这了,更多相关C++菱形继承内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 实例讲解C++编程中对设计模式中的原型模式的使用

    实例讲解C++编程中对设计模式中的原型模式的使用

    这篇文章主要介绍了C++编程中对设计模式中的原型模式的使用实例,包括原型模式中对C++的深拷贝和浅拷贝的处理,需要的朋友可以参考下
    2016-03-03
  • 使用C语言实现扫雷游戏

    使用C语言实现扫雷游戏

    这篇文章主要为大家详细介绍了使用C语言实现扫雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • C++中new与delete、malloc与free应用分析

    C++中new与delete、malloc与free应用分析

    这篇文章主要介绍了C++中new与delete、malloc与free应用分析,很重要的概念,需要的朋友可以参考下
    2014-08-08
  • C语言一级指针二级指针和三级指针区别及使用详解

    C语言一级指针二级指针和三级指针区别及使用详解

    这篇文章主要为大家介绍了C语言一级指针二级指针和三级指针,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • 用C++面向对象的方式动态加载so的方法

    用C++面向对象的方式动态加载so的方法

    下面小编就为大家带来一篇用C++面向对象的方式动态加载so的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • C++详细讲解互斥量与lock_guard类模板及死锁

    C++详细讲解互斥量与lock_guard类模板及死锁

    线程的主要优势在于,能够通过全局变量来共享信息。不过,这种便捷的共享是有代价的:必须确保多个线程不会同时修改同一变量,或者某一线程不会读取正由其他线程修改的变量。为了防止出现线程某甲试图访 问一共享变量时,线程某乙正在对其进行修改。引入了互斥量
    2022-07-07
  • C++基于boost asio实现sync tcp server通信流程详解

    C++基于boost asio实现sync tcp server通信流程详解

    这篇文章主要介绍了C++基于boost asio实现sync tcp server通信的流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • C++利用数组(一维/二维)处理批量数据的方法

    C++利用数组(一维/二维)处理批量数据的方法

    对于简单的问题,使用简单的数据类型就可以了,但是对于有些需要处理的数据,只用以上简单的数据类型是不够的,难以反映出数据的特点,也难以有效的进行处理,本文小编给大家介绍了C++利用数组(一维/二维)处理批量数据的方法,需要的朋友可以参考下
    2023-10-10
  • C++ 输入scanf()和输出printf()的操作

    C++ 输入scanf()和输出printf()的操作

    这篇文章主要介绍了C++ 输入scanf()和输出printf()的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • C语言实现linux网卡检测精简版

    C语言实现linux网卡检测精简版

    这篇文章主要为大家详细介绍了C语言实现linux网卡检测的精简版,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-06-06

最新评论