zoukankan      html  css  js  c++  java
  • 读书笔记_Effective_C++_条款四十:明智而审慎地使用多重继承

    多重继承是一种比较复杂的继承关系,它意味着如果用户想要使用这个类,那么就要对它的父类也了如指掌,所以在项目中会带来可读性的问题,一般我们都会尽量选择用单继承去替代它。

    使用多重继承过程容易碰到的问题就是名字冲突,像下面这样:

     1 class Base1
     2 {
     3 public:
     4     void fun(){}
     5 };
     6 
     7 class Base2
     8 {
     9 private:
    10     void fun(){}
    11 };
    12 
    13 class Derived : public Base1, public Base2
    14 {};
    15 
    16 int main()
    17 {
    18     Derived d;
    19     d.fun(); // error C2385: 对“fun”的访问不明确
    20     return 0;
    21 }

    因为在两个父类中都有名为fun的函数,所以这时候编译器不知道用户想调用的是哪个函数。但这里细心的读者会发现,这里我们是把Base2的fun的访问权限设为了private的。这个例子同时也说明了,编译器会优先去查找最合适的重载函数,再去考虑它的可访问性。如果真的要去访问重名的函数,可以指定作用域,像这样d.Base1::fun()(但注意d.Base2::fun()不行,因为它的访问性是private的)。

    多重继承另一个容易碰到的问题就是虚继承,我记得这还是面试官的一道面试题。试想一下,有一个父类名为A,类B和类C都继承于A,类D又同时继承了B和C(多重继承),那么如果不做任何处理,C++的类继承图里会包含两份A。

    但如果在继承的时候加了virtual,像下面这样:

    class B: virtual public A{…}
    class C: virtual pulibc A{…}

    那么D中就只有一份A了。C++标准库里面的流就是采用这样的形式,有一个父流basic_ios,basic_istream和basic_ostream分别虚继承于basic_ios,而basic_iostream又多重继承于basic_istream和basic_ostream。

    为了保证不会出现两份父类,只要是public继承理论上都应该有virutal关键字,但virutal也是有代价的,访问virtual base class的成员变量要比访问non-virutal base class的成员变量速度要慢。所以作者的忠告是:

    1. 非必要不使用virtual classes继承,普通情况请使用non-virtual classes继承

    2. 如果必须使用virtual base classes,尽可能避免在其中放置数据。

    后面的篇幅书上就举了一个多重继承的例子,在这里我就不说了,有兴趣的读者可以自己看看,但个人觉得还是能不用多重继承的时候,就尽量不用它,用复合+单继承往往能达到目的。

    最后总结一下:

    1. 多重继承比单一继承更复杂。它可能导致新的歧义性,以及对virtual继承的需要。

    2. virtual继承会增加大小、速度、初始化(及赋值)复杂度等等成本。如果virtual base classes不带任何数据,将是最具实用价值的情况。

    3. 多重继承的确有正当用途。其中一个情节涉及”public继承某个Interface class”和”private继承某个协助实现的class”的两两组合。

  • 相关阅读:
    HDU 4947 GCD Array 容斥原理+树状数组
    Codeforces Round #363 (Div. 2)
    白书 博弈学习
    2016 Multi-University Training Contest 1
    KM算法 PK 最小费用最大流
    final 发布 领跑衫获奖感想
    2016"百度之星"
    爆打团队 2016.05.05 站立会议
    爆打团队 四则运算 beta视频
    爆打团队 2016.04.27 站立会议
  • 原文地址:https://www.cnblogs.com/jerry19880126/p/3606577.html
Copyright © 2011-2022 走看看