派生类

来源:互联网 发布:js计算商品总金额 编辑:程序博客网 时间:2024/06/11 17:36

继承:它能够从已有的类派生出新的类,而派生类继承了原有类(称为基类)的特征。

下面地继承完成的一些工作:

  • 可以在已有类的基础上添加功能。
  • 可以给类添加数据。
  • 可以修改类方法的行为。

1 派生一个类

例如将RatedPlayer类声明为从TableTennisClass类派生而来:
class RatedPlayer:public TableTennisPlayer{   .......};

冒号指出RatedPlayer类的基类是TableTennisPlayer类。上述表明TableTennisPlayer是一个公有基类,这被称为公用派生。派生类对象包含基类对象。使用公用派生,基类的公用成员将成为派生类的公用成员;基类的私有部分也将成为派生类的一部分,但是只能通过基类的公用和包含方法访问。

RatedPlayer对象将具有以下特征:
1)派生类对象存储了基类的数据对象(派生类继承了基类的实现);
2)派生类对象可以使用基类的方法(派生类继承了基类的接口)。

需要在继承特性中添加什么呢?
1)派生类需要自己的构造函数;
2)派生类可以根据需要添加额外的数据成员和成员函数。

例如:
class RatedPlayer:public TableTennisPlayer{   private:      unsigned int rating;   public:      RatedPlayer(unsigned int r=0,const string &fn="none",const string &ln="none",bool ht=false);RatedPlayer(unsigned int r,const TableTennisPlayer &tp);unsigned int Rating() const{return rating;}void ResetRating(unsigned int r){rating=r;}};

2 构造函数:访问权限的考虑

派生类不能直接访问基类的私有成员,而必须通过基类方法进行访问。例如,RatedPlayer构造函数不能直接设置继承的成员,而必须使用基类的公用方法来访问私有的基类成员。具体地说,派生类构造函数必须使用基类构造函数。
创建派生类对象时,首先创建基类的对象。从概念上说,这意味着基类对象应当在程序进入派生类构造函数之前被创建。C++使用成员初始化列表语法来完成这种工作。例如,下面第一个构造函数的代码:

RatedPlayer::RatedPlayer(unsigned int r,const string &fn,const string &ln,bool ht):TableTennisPlayer(fn,ln,ht){    rating=r;}

其中:TableTennisPlayer(fn,ln,ht)是成员初始化列表。它是可执行的代码,调用TableTennisPlayer构造函数。

如果省略成员初始化列表,情况将如何呢?

必须首先创建基类的对象,如果不调用基类的构造函数,程序将使用默认的基类构造函数,因此没有初始化列表等同于下面情况:
RatedPlayer::RatedPlayer(unsigned int r,const string &fn,const string &ln,bool ht):TableTennisPlayer(){    rating=r;}

对于第二个构造函数的代码:
RatedPlayer::RatedPlayer(unsigned int r,const TableTennisPlayer &tp):TableTennisPlayer(tp){    rating=r;}

这里将TableTennisPlayer的信息传递给了TableTennisPlayer构造函数:
TableTennisPlayer(tp);
由于tp的类型为TableTennisPlayer&,因此将调用基类的复制构造函数。基类没有定义复制构造函数,如果需要使用复制构造函数但又没有定义,编译器将自动生成一个。在这种情况下,执行成员复制的隐式复制构造函数是合适的,因为这个类没有使用动态内存分配。
如果愿意,也可以对派生类成员使用成员初始化列表语法。在这种情况下,应在列表中使用成员名。而不是类名。所以,第二个构造函数可以按照下述方式编写:
RatedPlayer::RatedPlayer(unsigned int r,const TableTennisPlayer &tp):TableTennisPlayer(tp),rating(r){    }

有关派生类的构造函数的要点如下:
1)首先创建基类对象;
2)派生类构造函数应通过成员初始化列表将基类信息传递给基类的构造函数;
3)派生类构造函数应初始化派生类新增的数据成员。

释放对象的顺序与创建对象的顺序相反,即首先执行派生类的析构函数,然后自动调用基类的析构函数。

注意:创建派生类对象时,程序首先调用基类构造函数,然后再调用派生类构造函数。基类构造函数负责初始化继承的数据成员;派生类构造函数注意用于初始化新增的数据成员。派生类的构造函数总是调用一个基类构造函数。可以使用初始化列表语法指明要使用的基类构造函数,否则将使用默认的基类构造函数。

派生类对象过期时,程序将首先调用派生类析构函数,然后再调用基类析构函数。

成员初始化列表:

派生类构造函数可以使用初始化列表机制将值传递给基类的构造函数。例如:

derived::deriver(type1 x,type2 y):base(x,y){   .....}

其中derived是派生类,base是基类,x和y是基类构造函数使用的变量。例如,如果派生类构造函数接收到参数,这种机制将把参数传递给被定义为接受这些类型的基类构造函数。除虚基类外,类只能将值传递会相邻的基类,但后者可以使用相同的机制将信息传递给相邻的基类,以此类推。如果没有成员初始化列表中提供基类构造函数,程序将使用默认的基类构造函数。成员初始化列表只能用于构造函数。

3派生类和基类的关系

派生类与基类之间有一些特殊关系。其中之一是派生类对象可以使用基类的方法,条件是方法不是私有的:
另外两个重要的关系式:基类指针可以在不进行显式类型转换的情况下指向派生类对象;基类引用可以在不进行显示类型转换的情况下引用派生类的对象:

然而,基类的指针和引用只能用于调用基类的方法,因此,不能使用rt或pt来调用只存在于派生类中新增的方法。
通常,C++要求引用和指针类型与赋给的类型匹配,但这一规则对继承来说是例外。然而,这种例外只是单向的,不可以将基类对象和地址赋给派生类的引用和指针;


0 0
原创粉丝点击