C++之虚拟继承

断了今生、忘了曾经 提交于 2020-01-22 21:30:52

定义:  
  class   B  
  {  
  private:  
          B()  
          {  
          }  
  friend   class   A;  
  };  
  将导致B无法被除A以外的其它任何class直接继承以后实例化,也就是说,在上面这个定义的基础上,如果你再定义:  
          class   C:public   B{};  
  将导致编译能够通过,但是无法实例化C(那当然也没用了,所以间接实现了一个无法继承的类B),但是因为A是B的友元,所以能够进入B的private区域,所以如果定义:  
          class   A:public   B{};  
  能够实例化A.  
  但是这样定义还有一个漏洞,如果在A普通public继承B的基础上再定义:  
          class   D:public   A{};  
  你会发现D也是可以实例化的,那么相当于间接public继承了B。这显然不是我们想要的,所以,正确的做法是(完整代码):  
  class   A;  
  class   B  
  {  
  private:  
          B()  
          {  
          }  
  friend   class   A;  
  };  
  class   A:virtual   public   B{};   //A   is   the   one   we're   looking   for  
  也就是说要做到三步:  
  1)将你B的相关构造函数放入private区域  
  2)声明子类A为B的友元  
  3)定义A为virtual   public继承B  
  这样创建的代码的效果是:  
  B不能被继承也不能被实例化  
  A可以被实例化,但是不能被继承 

====================================  
  确切地说,A可以被继承,但是继承了A的类不能被实例化(那这个类也就没用了)。假设有  
        class   E:public   A{};  
  则实例化E时:  
        E   e;  
  系统将报告无法进入B的私有成员声明而无法通过编译。  
  这是因为   A是   virtual   继承   B的,所以,E再继承A的时候,需要由E去调用B的构造函数,但是E不是B的友元,所以无法编译通过  
  但是如果A不是virtual继承B的,那么E再继承A时,构造函数的调用情况是E调用父类A的构造函数,A再调用A的父类B的构造函数,而A是其父类B的友元,所以这一串调用能够顺利进行。

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!