C++:多重继承

简介:

单继承或单基派生:派生类只有一个基类;
多继承或多基派生:派生类有两个或两个以上的基类。

4.4.1 多重继承派生类的声明
在C++中,声明具有两个以上的派生类与声明单机派生类的形式相似,只需将要继承的多个基类用逗号分分隔即可,其声明的一般形式如下:

class 派生类名:继承方式1 基类名,...,继承方式n 基类名{
           派生类新增的数据成员和成员函数 
};

冒号后面的部分称为基类表,各基类之间用逗号分隔,其中“继承方式i”(i=1,2,...,n),规定了派生类从基类中按什么方式继承:private、protected、public。默认的继承方式是private。

例如:

     class z:public x,y{ //类z公有继承了类x,私有继承了y(默认的) 
                 ...
     };
     class z:x,public y{ //类z私有继承了类x(默认的),公有继承了y
                  ...
     };
    class z:public x,public y{ //类z公有继承了类x和类y 
                  ...
      }; 

在多种继承中,三种继承方式对于基类成员在派生类中的访问属性规则与单继承相同。

//例4.13 多重继承情况下派生类的访问特性。

复制代码
#include<iostream>
using namespace std;
class X{                     //声明基类X 
   public:
     void setX(int x)
     {
      a = x;
     }
     void showX()
     {
      cout<<"a="<<a<<endl;
     }
   private:
     int a; 
};
class Y{                     //声明基类Y 
   public:
     void setY(int y)
     {
      b = y;
     }
     void showY()
     {
      cout<<"b="<<b<<endl;
     }
   private:
     int b; 
};
class Z:public X,private Y{    //声明派生类Z,公有继承了类X,私有继承了类Y 
   public:
     void setZ(int x,int y)
     {
      c = y;
      setY(x);
     }
     void showZ()
     {
      showY();
      cout<<"c="<<c<<endl;
     }
   private:
     int c; 
};
int main()
{
 Z obj; 
 obj.setX(3);        //正确,成员函数setX在Z中仍是公有成员 
 obj.showX();        //正确,成员函数showX在Z中仍是公有成员 
 //obj.setY(3);      //错误,成员函数setY在Z中成为私有成员 
 //obj.showY();      //错误,成员函数setY在Z中成为私有成员  
 obj.setZ(6,8);
 obj.showZ();
 return 0; 
} 
复制代码

说明:对基类成员的访问必须是无二义的,例如下列程序段对基类成员的访问是二义的,必须想办法消除二义性。

复制代码
  class X{
        public:
          int f();
      }; 
      class Y{
        publicint f();
          int g(); 
      };
      class Z:public X,public Y{
        public:
          int g();
          int h();
      };
复制代码

如定义类Z的对象obj:
      Z obj;
则以下对函数f()的访问是二义性的:
      obj.f(); //二义性错误,不知调用的是类X的f(),还是类Y的f()
使用成员名限定可以消除二义性,例如:
      obj.X::f(); //调用了类X的f() 
      obj.Y::f(); //调用了类Y的f() 

 

4.4.2 多重继承派生类构造函数和析构函数
多重继承下派生类构造函数的定义形式与单继承派生类构造函数的定义形式相似,只是n个基类的构造函数之间用逗号分隔。多重继承下派生类构造函数定义的一般形式如下:


派生类名(参数总表):基类名1(参数表1),基类名2(参数表2),...,基类名n(参数表n)
{
        派生类新增成员的初始化语句 
}

多重继承的构造函数的执行顺序与单继承构造函数的执行顺序相同,也是遵循先执行基类的构造函数,在执行对象成员的构造函数,最后执行派生类的构造函数体的原则。处于同一层次的各个基类构造函数的执行顺序,取决于声明派生所指定的各个基类的顺序,与派生类构造函数中所定义的成员初始化列表的各项顺序没有关系。由于析构函数不带参数,在派生类中是否要定义析构函数与它所属的基类无关,所以与单继承情况类似,基类的析构函数不会因为派生类没有析构函数而得不到执行,它们是各自独立的。析构函数的执行顺序与构造函数的执行顺序相反。

 例如,由一个硬件类Hard和一个软件类Soft共同派生出计算机系统类System,声明如下:
复制代码
class Hard{                   //声明硬件类Hard 
      protected:
        char bodyname[20];
      public:
        Hard(char* bn);         //基类Hard的构造函数 
        ...  
    };
    class Soft{                   //声明软件类soft
      protected:
        char os[10];
        char Lang[15];
      public:
        Soft(char* o,char* lg);    //基类Soft的构造函数 
        ...  
    };
    class System:public Hard,public Soft{     //声明计算机系统类System(派生类) 
      private:
        char owner[10];
      public:
        System(char* ow,char* bn,char* o,char* lg):Hard(bn),Soft(o,lg);
        ...                              //派生类System的构造函数,缀上了基类Hard和Soft的构造函数     
    };
    注意:这是一段示意性的程序段,表示在定义计算机系统派生类System的构造函数时,缀上了硬件基类Hard
和软件基类Soft的构造函数。
  再如,现有一个窗口类Window和一个滚动条类Scrollbar,它们可以共同派生出一个带有滚动条的窗口,声明如下:
      class Window{                //声明窗口类 
       ...
       public:
         Window(int top,int left,int bottom,int right);
         ~Window();
         ... 
    };
    class Scrollbar{             //声明滚动条类 
       ...
       public:
          Scrollbar(int top,int left,int bottom,int right);
          ~Scrollbar();
          ... 
    };
    class ScrollbarWind:Window,Scrollbar{            //声明带有滚动条的窗口派生类 
       ...
       Scrollbar(int top,int left,int bottom,int right);
       ~Scrollbar();
       ... 
    };
    Scrollbar::Scrollbar(int top,int left,int bottom,int right)
    :Window(top,left,bottom,right),Scrollbar(top,right-20,bottom,right)
    {
       ...
    }
    这也是一段示意性的程序大段,表示定义带有滚动条窗口派生类ScrollbarWind的构造
    函数时,也缀上了窗口基类Window和滚动条基类Scrollbar的构造函数。
复制代码

例4.14 多重继承情况下派生类构造函数和析构函数的定义方法。

复制代码
#include<iostream>
using namespace std;
class X{
  public:
    X(int sa)           //基类X的构造函数 
    {
     a = sa;
     cout<<"X_Construction called."<<endl;
    }
    int getX()
    {
      return a;
    }
    ~X()                //基类X的析构函数  
    {
      cout<<"X_Destruction called."<<endl;
    }
  private:
    int a;    
};
class Y{
  public:
    Y(int sb)           //基类Y的构造函数 
    {
     b = sb;
     cout<<"Y_Construction called."<<endl;
    }
    int getY()
    {
      return b;
    }
    ~Y()                //基类Y的析构函数  
    {
      cout<<"Y_Destruction called."<<endl;
    }
  private:
    int b;    
};
class Z:public X,private Y{
  public:
    Z(int sa,int sb,int sc):X(sa),Y(sb)
    {
     c = sc;
     cout<<"Z_Construction called."<<endl;
    }
    int getZ()
    {
      return c;
    }
    Y::getY;                //访问声明 
    //int getY()            //覆盖基类Y的getY成员函数 
    //{
    //  return Y::getY();
    //}
    ~Z()
    {
      cout<<"Z_Destruction called."<<endl;
    }
  private:
    int c;
};
int main()
{
  Z obj(2,4,6);
  int ma=obj.getX();
  cout<<"a="<<ma<<endl;
  int mb=obj.getY();
  cout<<"b="<<mb<<endl;
  int mc=obj.getZ();
  cout<<"c="<<mc<<endl;
  return 0;  
}
/*
 运行结果是:
                 X_Construction called.
                 Y_Construction called.
                 Z_Construction called.
                 a=2
                 b=4
                 c=6
                 Z_Destruction called.
                 Y_Destruction called.
                 X_Destruction called.
*/
复制代码

 

程序猿神奇的手,每时每刻,这双手都在改变着世界的交互方式!

本文转自当天真遇到现实博客园博客,原文链接:http://www.cnblogs.com/XYQ-208910/p/4912406.html ,如需转载请自行联系原作者
相关文章
|
8月前
|
C++
C++程序中的多重继承
C++程序中的多重继承
85 1
|
6月前
|
编译器 C++ 开发者
C++一分钟之-多重继承与菱形问题
【7月更文挑战第19天】C++的多重继承允许类从多个基类派生,但引入了菱形问题,即类D通过B和C(都继承自A)双重继承A,可能导致数据冗余和二义性。解决这个问题的关键是**虚继承**,通过`virtual`关键字确保基类A只被继承一次,消除冲突。理解并适当使用虚继承是处理这类问题的关键,有助于保持代码的清晰和正确性。
106 0
|
8月前
|
设计模式 编译器 数据安全/隐私保护
C++ 多级继承与多重继承:代码组织与灵活性的平衡
C++的多级和多重继承允许类从多个基类继承,促进代码重用和组织。优点包括代码效率和灵活性,但复杂性、菱形继承问题(导致命名冲突和歧义)以及对基类修改的脆弱性是潜在缺点。建议使用接口继承或组合来避免菱形继承。访问控制规则遵循公有、私有和受保护继承的原则。在使用这些继承形式时,需谨慎权衡优缺点。
183 1
|
8月前
|
C++
[C++/PTA] 日程安排(多重继承+重载)
[C++/PTA] 日程安排(多重继承+重载)
199 0
|
Java 编译器 PHP
C++的多重继承
派生类都只有一个基类,称为单继承(Single Inheritance)。除此之外,C++也支持多继承(Multiple Inheritance),即一个派生类可以有两个或多个基类。 多继承容易让代码逻辑复杂、思路混乱,一直备受争议,中小型项目中较少使用,后来的 Java、C#、PHP 等干脆取消了多继承。 多继承的语法也很简单,将多个基类用逗号隔开即可。例如已声明了类A、类B和类C,那么可以这样来声明派生类D: class D: public A, private B, protected C{ //类D新增加的成员 } D 是多继承形式的派生类,它以公有的方式继承 A 类,
|
编译器 C++
C++中的多重继承
🐰多重继承 🌸声明多重继承的方法 🌸多重继承派生类的构造函数与析构函数 🌸多重继承引起的二义性
|
Java 编译器 Android开发
Android C++系列:C++最佳实践4多重继承与虚继承
Java和C++在语法层面比较的时候就不得不提到C++的多继承,我们知道Android是单继承,C++是多继承。在大型项目中不可避免的会用到多继承,本文分析C++多继承的一些特征。
148 0
|
Java C++
C++继承与派生解析(继承、重载/转换运算符、多重继承、多态、虚函数/纯虚函数、抽象类)
C++继承与派生解析(继承、重载/转换运算符、多重继承、多态、虚函数/纯虚函数、抽象类)
215 0
C/C++---Person类、学生类、教师类和研究生类(多重继承)
C/C++---Person类、学生类、教师类和研究生类(多重继承)
921 0