(C++-面向对象程序设计-谭浩强)第11章-继承与派生(共37页).doc

上传人:飞****2 文档编号:14097882 上传时间:2022-05-02 格式:DOC 页数:37 大小:172.50KB
返回 下载 相关 举报
(C++-面向对象程序设计-谭浩强)第11章-继承与派生(共37页).doc_第1页
第1页 / 共37页
(C++-面向对象程序设计-谭浩强)第11章-继承与派生(共37页).doc_第2页
第2页 / 共37页
点击查看更多>>
资源描述

《(C++-面向对象程序设计-谭浩强)第11章-继承与派生(共37页).doc》由会员分享,可在线阅读,更多相关《(C++-面向对象程序设计-谭浩强)第11章-继承与派生(共37页).doc(37页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。

1、精选优质文档-倾情为你奉上第4篇 面向对象的程序设计第11章 继承与派生第12章 多态性与虚函数第13章 输入输出流第14章 C+工具第11章 继承与派生11.1 继承与派生的概念11.2 派生类的声明方式11.3 派生类的构成11.4 派生类成员的访问属性11.5 派生类的构造函数和析构函数11.6 多重继承11.7 基类与派生类的转换11.8 继承与组合11.9 继承在软件开发中的重要意义面向对象程序设计有4个主要特点:抽象、封装、继承和多态性。要较好地进行面向对象程序设计,还必须了解面向对象程序设计另外两个重要特征继承性和多态性。在本章中主要介绍有关继承的知识,在第12章中将介绍多态性。

2、面向对象技术强调软件的可重用性(software reusability)。C+语言提供了类的继承机制,解决了软件重用问题。11.1 继承与派生的概念在C+中可重用性是通过继承(inheritance)这一机制来实现的。继承是C+的一个重要组成部分。一个类中包含了若干数据成员和成员函数。在不同的类中,数据成员和成员函数是不相同的。但有时两个类的内容基本相同或有一部分相同。利用原来声明的类Student作为基础,再加上新的内容即可,以减少重复的工作量。C+提供的继承机制就是为了解决这个问题。在第8章已举了马的例子来说明继承的概念。见图11.1示意。在C+中,所谓“继承”就是在一个已存在的类的基础

3、上建立一个新的类。已存在的类(例如“马”)称为“基类(base class )”或“父类(father class )”。新建的类(例如“公马”)称为“派生类(derived class )”或“子类(son class )”。见图11.2示意。图11.1图11.2一个新类从已有的类那里获得其已有特性,这种现象称为类的继承。通过继承,一个新建子类从已有的父类那里获得父类的特性。从另一角度说,从已有的类(父类)产生一个新的子类,称为类的派生。类的继承是用已有的类来建立专用类的编程技术。派生类继承了基类的所有数据成员和成员函数,并可以对成员作必要的增加或调整。一个基类可以派生出多个派生类,每一个派

4、生类又可以作为基类再派生出新的派生类,因此基类和派生类是相对而言的。以上介绍的是最简单的情况:一个派生类只从一个基类派生,这称为单继承(single inheritance),这种继承关系所形成的层次是一个树形结构,可以用图11.3表示。图11.3请注意图中箭头的方向,在本书中约定,箭头表示继承的方向,从派生类指向基类。一个派生类不仅可以从一个基类派生,也可以从多个基类派生。一个派生类有两个或多个基类的称为多重继承(multiple inheritance), 这种继承关系所形成的结构如图11.4所示。图11.4关于基类和派生类的关系,可以表述为:派生类是基类的具体化,而基类则是派生类的抽象。

5、图11.511.2 派生类的声明方式假设已经声明了一个基类Student,在此基础上通过单继承建立一个派生类Student1:class Student1: public Student/声明基类是Studentpublic:void display_1( )/新增加的成员函数coutage: ageendl;coutaddress: addrnumnamesex;void display( )cout num: numendl;cout name: nameendl;cout sex: sexendl;private : /基类私有成员int num; string name;char se

6、x;class Student1: public Student /以public方式声明派生类Student1public:void display_1( )cout num: numendl; /企图引用基类的私有成员,错误cout name: nameendl; /企图引用基类的私有成员,错误cout sex: sexendl; /企图引用基类的私有成员,错误cout age: ageendl; /引用派生类的私有成员,正确cout address: addrendl; /引用派生类的私有成员,正确private: int age;string addr;由于基类的私有成员对派生类来说是

7、不可访问的,因此在派生类中的display_1函数中直接引用基类的私有数据成员num,name和sex是不允许的。只能通过基类的公用成员函数来引用基类的私有数据成员。可以将派生类Student1的声明改为class Student1: public Student/以public方式声明派生类Student1public:void display_1( ) cout age: ageendl; /引用派生类的私有成员,正确cout address: addrendl; /引用派生类的私有成员,正确private:int age; string addr;然后在main函数中分别调用基类的dis

8、play函数和派生类中的display_1函数,先后输出5个数据。可以这样写main函数(假设对象stud中已有数据):int main( )Student1 stud;/定义派生类Student1的对象studstud.display( );/调用基类的公用成员函数,输出基类中3个数据成员的值stud.display_1();/调用派生类公用成员函数,输出派生类中两个数据成员的值return 0;请根据上面的分析,写出完整的程序,程序中应包括输入数据的函数。实际上,程序还可以改进,在派生类的display_1函数中调用基类的display函数,在主函数中只要写一行:stud.display_

9、1(); 即可输出5个数据。11.4.2 私有继承在声明一个派生类时将基类的继承方式指定为private的,称为私有继承,用私有继承方式建立的派生类称为私有派生类(private derived class ) , 其基类称为私有基类(private base class )。私有基类的公用成员和保护成员在派生类中的访问属性相当于派生类中的私有成员,即派生类的成员函数能访问它们,而在派生类外不能访问它们。私有基类的私有成员在派生类中成为不可访问的成员,只有基类的成员函数可以引用它们。一个基类成员在基类中的访问属性和在派生类中的访问属性可能是不同的。私有基类的成员在私有派生类中的访问属性见书中表

10、11.2。图11.7图11.7表示了各成员在派生类中的访问属性。对表11.2的规定不必死记,只需理解: 既然声明为私有继承,就表示将原来能被外界引用的成员隐藏起来,不让外界引用,因此私有基类的公用成员和保护成员理所当然地成为派生类中的私有成员。私有基类的私有成员按规定只能被基类的成员函数引用,在基类外当然不能访问他们,因此它们在派生类中是隐蔽的,不可访问的。对于不需要再往下继承的类的功能可以用私有继承方式把它隐蔽起来,这样,下一层的派生类无法访问它的任何成员。可以知道: 一个成员在不同的派生层次中的访问属性可能是不同的。它与继承方式有关。例11.2 将例11.1中的公用继承方式改为用私有继承方

11、式(基类Student不改)。可以写出私有派生类如下:class Student1: private Student/用私有继承方式声明派生类Student1public:void display_1( ) /输出两个数据成员的值coutage: ageendl; /引用派生类的私有成员,正确coutaddress: addrendl; /引用派生类的私有成员,正确private:int age; string addr;请分析下面的主函数:int main( )Student1 stud1;/定义一个Student1类的对象stud1stud1.display(); /错误,私有基类的公用成

12、员函数在派生类中是私有函数stud1.display_1( );/正确。Display_1函数是Student1类的公用函数stud1.age=18; /错误。外界不能引用派生类的私有成员return 0;可以看到:(1) 不能通过派生类对象(如stud1)引用从私有基类继承过来的任何成员(如stud1.display()或stud1.num)。(2) 派生类的成员函数不能访问私有基类的私有成员,但可以访问私有基类的公用成员(如stud1.display_1函数可以调用基类的公用成员函数display,但不能引用基类的私有成员num)。有没有办法调用私有基类的公用成员函数,从而引用私有基类的私

13、有成员呢?有。应当注意到: 虽然在派生类外不能通过派生类对象调用私有基类的公用成员函数,但可以通过派生类的成员函数调用私有基类的公用成员函数(此时它是派生类中的私有成员函数,可以被派生类的任何成员函数调用)。可将上面的私有派生类的成员函数定义改写为void display_1( )/输出5个数据成员的值display(): /调用基类的公用成员函数,输出3个数据成员的值coutage: ageendl; /输出派生类的私有数据成员coutaddress: addrendl; /输出派生类的私有数据成员main函数可改写为int main( )Student1 stud1;stud1.displ

14、ay_1( );/display_1函数是派生类Student1类的公用函数return 0;这样就能正确地引用私有基类的私有成员。可以看到,本例采用的方法是: 在main函数中调用派生类中的公用成员函数stud1.display_1; 通过该公用成员函数调用基类的公用成员函数display(它在派生类中是私有函数,可以被派生类中的任何成员函数调用); 通过基类的公用成员函数display引用基类中的数据成员。请根据上面的要求,补充和完善上面的程序,使之成为完整、正确的程序。程序中应包括输入数据的函数。由于私有派生类限制太多,使用不方便,一般不常使用。11.4.3 保护成员和保护继承由prot

15、ected声明的成员称为“受保护的成员”,或简称“保护成员”。从类的用户角度来看,保护成员等价于私有成员。但有一点与私有成员不同,保护成员可以被派生类的成员函数引用。见图11.8示意。图11.8如果基类声明了私有成员,那么任何派生类都是不能访问它们的,若希望在派生类中能访问它们,应当把它们声明为保护成员。如果在一个类中声明了保护成员,就意味着该类可能要用作基类,在它的派生类中会访问这些成员。在定义一个派生类时将基类的继承方式指定为protected的,称为保护继承,用保护继承方式建立的派生类称为保护派生类(protected derived class ), 其基类称为受保护的基类(prote

16、cted base class ),简称保护基类。保护继承的特点是: 保护基类的公用成员和保护成员在派生类中都成了保护成员,其私有成员仍为基类私有。也就是把基类原有的公用成员也保护起来,不让类外任意访问。将表11.1和表11.2综合起来,并增加保护继承的内容,见书中表11.3。保护基类的所有成员在派生类中都被保护起来,类外不能访问,其公用成员和保护成员可以被其派生类的成员函数访问。比较一下私有继承和保护继承(也就是比较在私有派生类中和在保护派生类中的访问属性), 可以发现,在直接派生类中,以上两种继承方式的作用实际上是相同的: 在类外不能访问任何成员,而在派生类中可以通过成员函数访问基类中的公

17、用成员和保护成员。但是如果继续派生,在新的派生类中,两种继承方式的作用就不同了。例如,如果以公用继承方式派生出一个新派生类,原来私有基类中的成员在新派生类中都成为不可访问的成员,无论在派生类内或外都不能访问,而原来保护基类中的公用成员和保护成员在新派生类中为保护成员,可以被新派生类的成员函数访问。从表11.3可知: 基类的私有成员被派生类继承后变为不可访问的成员,派生类中的一切成员均无法访问它们。如果需要在派生类中引用基类的某些成员,应当将基类的这些成员声明为protected,而不要声明为private。如果善于利用保护成员,可以在类的层次结构中找到数据共享与成员隐蔽之间的结合点。既可实现某

18、些成员的隐蔽,又可方便地继承,能实现代码重用与扩充。通过以上的介绍,可以知道:(1) 在派生类中,成员有4种不同的访问属性: 公用的,派生类内和派生类外都可以访问。 受保护的,派生类内可以访问,派生类外不能访问,其下一层的派生类可以访问。 私有的,派生类内可以访问,派生类外不能访问。 不可访问的,派生类内和派生类外都不能访问。可以用书中表11.4表示。需要说明的是: 这里所列出的成员的访问属性是指在派生类中所获得的访问属性。 所谓在派生类外部,是指在建立派生类对象的模块中,在派生类范围之外。 如果本派生类继续派生,则在不同的继承方式下,成员所获得的访问属性是不同的,在本表中只列出在下一层公用派

19、生类中的情况,如果是私有继承或保护继承,读者可以从表11.3中找到答案。(2) 类的成员在不同作用域中有不同的访问属性,对这一点要十分清楚。下面通过一个例子说明怎样访问保护成员。例11.3 在派生类中引用保护成员。#include #include using namespace std;class Student/声明基类public : /基类公用成员void display( );protected:/基类保护成员int num;string name;char sex;/定义基类成员函数void Student:display( ) coutnum: numendl;coutname:

20、 nameendl;coutsex: sexendl;class Student1: protectedStudent /用protected方式声明派生类Student1public : void display1( );/派生类公用成员函数private : int age;/派生类私有数据成员string addr;/派生类私有数据成员;void Student1:display1( )/定义派生类公用成员函数coutnum: numendl;/引用基类的保护成员,合法coutname: nameendl;/引用基类的保护成员,合法coutsex: sexendl;/引用基类的保护成员,

21、合法coutage: ageendl;/引用派生类的私有成员,合法coutaddress: addrendl; /引用派生类的私有成员,合法int main( )Student1 stud1; /stud1是派生类Student1类的对象stud1.display1( ); /合法,display1是派生类中的公用成员函数stud1.num=10023; /错误,外界不能访问保护成员return 0;在派生类的成员函数中引用基类的保护成员是合法的。保护成员和私有成员不同之处,在于把保护成员的访问范围扩展到派生类中。注意: 在程序中通过派生类Student1的对象stud1的公用成员函数disp

22、lay1去访问基类的保护成员num.name和sex,不要误认为可以通过派生类对象名去访问基类的保护成员。请补充、修改上面的程序,使之能正常运行。程序中应包括输入数据的函数。私有继承和保护继承方式在使用时需要十分小心,很容易搞错,一般不常用。11.4.4 多级派生时的访问属性图11.9如果有图11.9所示的派生关系: 类A为基类,类B是类A的派生类,类C是类B的派生类,则类C也是类A的派生类。类B称为类A的直接派生类,类C称为类A的间接派生类。类A是类B的直接基类,是类C的间接基类。在多级派生的情况下,各成员的访问属性仍按以上原则确定。例11.4 多级派生的访问属性。如果声明了以下的类: cl

23、ass A /基类public:int i;protected :void f2( );int j;private :int k;class B: public A/public方式public :void f3( );protected :void f4( );private :int m;class C: protected B /protected方式public :void f5( );private :int n;类A是类B的公用基类,类B是类C的保护基类。各成员在不同类中的访问属性如下: 无论哪一种继承方式,在派生类中是不能访问基类的私有成员的,私有成员只能被本类的成员函数所访问,

24、毕竟派生类与基类不是同一个类。如果在多级派生时都采用公用继承方式,那么直到最后一级派生类都能访问基类的公用成员和保护成员。如果采用私有继承方式,经过若干次派生之后,基类的所有的成员已经变成不可访问的了。如果采用保护继承方式,在派生类外是无法访问派生类中的任何成员的。而且经过多次派生后,人们很难清楚地记住哪些成员可以访问,哪些成员不能访问,很容易出错。因此,在实际中,常用的是公用继承。11.5 派生类的构造函数和析构函数用户在声明类时可以不定义构造函数,系统会自动设置一个默认的构造函数,在定义类对象时会自动调用这个默认的构造函数。这个构造函数实际上是一个空函数,不执行任何操作。如果需要对类中的数

25、据成员初始化,应自己定义构造函数。构造函数的主要作用是对数据成员初始化。在设计派生类的构造函数时,不仅要考虑派生类所增加的数据成员的初始化,还应当考虑基类的数据成员初始化。也就是说,希望在执行派生类的构造函数时,使派生类的数据成员和基类的数据成员同时都被初始化。解决这个问题的思路是: 在执行派生类的构造函数时,调用基类的构造函数。11.5.1 简单的派生类的构造函数任何派生类都包含基类的成员,简单的派生类只有一个基类,而且只有一级派生(只有直接派生类,没有间接派生类),在派生类的数据成员中不包含基类的对象(即子对象)。例11.5 简单的派生类的构造函数。#include #includeusi

26、ng namespace std;class Student/声明基类Studentpublic:Student(int n,string nam,char s) /基类构造函数num=n; name=nam; sex=s; Student( ) /基类析构函数protected : /保护部分int num;string name;char sex ;class Student1: public Student /声明派生类Student1public : /派生类的公用部分Student1(int n,string nam,char s,int a,string ad):Student(n

27、,nam,s)/派生类构造函数age=a; /在函数体中只对派生类新增的数据成员初始化addr=ad; void show( )coutnum: numendl; coutname: nameendl; coutsex: sexendl; coutage: ageendl; coutaddress: addrendlendl;Student1( ) /派生类析构函数private : /派生类的私有部分int age;string addr;int main( )Student1 stud1(10010,Wang-li,f,19,115 Beijing Road,Shanghai); Stud

28、ent1 stud2(10011,Zhang-fun,m,21,213 Shanghai Road,Beijing); stud1.show( ); /输出第一个学生的数据 stud2.show( ); /输出第二个学生的数据return 0;运行结果为num:10010name:Wang-lisex:faddress: 115Beijing Road,Shanghainum:10011name:Zhang-funsex:maddress: 213Shanghai Road,Beijing请注意派生类构造函数首行的写法: Student1(int n, string nam, char s,

29、int a, string ad):Student(n, nam, s)其一般形式为派生类构造函数名(总参数表列):基类构造函数名(参数表列)派生类中新增数据成员初始化语句在main函数中,建立对象stud1时指定了5个实参。它们按顺序传递给派生类构造函数Student1的形参。然后,派生类构造函数将前面3个传递给基类构造函数的形参。见图11.10。图11.10通过Student (n, nam, s)把3个值再传给基类构造函数的形参,见图11.11。Student( n, nam, s) Student(int n, string nam, char s)/这是基类构造函数的首部图11.11

30、在上例中也可以将派生类构造函数在类外面定义,而在类体中只写该函数的声明: Student1(int n, string nam, char s, int a, string ad);在类的外面定义派生类构造函数:Student1Student1(int n,string nam,char s,int a,string ad):Student(n,nam,s)age=a;addr=ad; 请注意: 在类中对派生类构造函数作声明时,不包括基类构造函数名及其参数表列(即Student(n, nam, s))。只在定义函数时才将它列出。在以上的例子中,调用基类构造函数时的实参是从派生类构造函数的总参数

31、表中得到的,也可以不从派生类构造函数的总参数表中传递过来,而直接使用常量或全局变量。例如,派生类构造函数首行可以写成以下形式:Student1(string nam,char s,int a,string ad):Student(10010,nam,s)即基类构造函数3个实参中,有一个是常量10010,另外两个从派生类构造函数的总参数表传递过来。请回顾一下在第9章介绍过的构造函数初始化表的例子:Box:Box(int h,int w,int len):height(h),width(w),length(len) 它也有一个冒号,在冒号后面的是对数据成员的初始化表。实际上,本章介绍的在派生类构造

32、函数中对基类成员初始化,就是在第9章介绍的构造函数初始化表。也就是说,不仅可以利用初始化表对构造函数的数据成员初始化,而且可以利用初始化表调用派生类的基类构造函数,实现对基类数据成员的初始化。也可以在同一个构造函数的定义中同时实现这两种功能。例如,例11.5中派生类的基类构造函数的定义采用了下面的形式:Student1(int n, string nam,char s,int a, string ad):Student(n,nam,s)age=a;/在函数体中对派生类数据成员初始化 addr=ad;可以将对age和addr的初始化也用初始化表处理,将构造函数改写为以下形式: Student1(

33、int n, string nam,char s,int a, string ad):Student(n,nam,s),age(a),addr(ad) 这样函数体为空,更显得简单和方便。在建立一个对象时,执行构造函数的顺序是:派生类构造函数先调用基类构造函数;再执行派生类构造函数本身(即派生类构造函数的函数体)。对上例来说,先初始化num,name,sex,然后再初始化age和addr。在派生类对象释放时,先执行派生类析构函数Student1( ),再执行其基类析构函数Student( )。11.5.2 有子对象的派生类的构造函数类的数据成员中还可以包含类对象,如可以在声明一个类时包含这样的数

34、据成员:Student s1;/ Student是已声明的类名,s1是Student类的对象这时,s1就是类对象中的内嵌对象,称为子对象(subobject),即对象中的对象。通过例子来说明问题。例11.6 包含子对象的派生类的构造函数。为了简化程序以易于阅读,这里设基类Student的数据成员只有两个,即num和name。#include #include using namespace std;class Student/声明基类public : /公用部分Student(int n, string nam ) /基类构造函数,与例11.5相同num=n;name=nam; void di

35、splay( ) /成员函数,输出基类数据成员coutnum:numendlname:nameendl;protected : /保护部分int num;string name;class Student1: public Student /声明公用派生类Student1public:Student1(int n, string nam,int n1, string nam1,int a, string ad):Student(n,nam),monitor(n1,nam1) /派生类构造函数age=a; addr=ad;void show( )coutThis student is:endl;

36、display(); /输出num和name coutage: ageendl; /输出age coutaddress: addrendlendl; /输出addrvoid show_monitor( ) /成员函数,输出子对象coutendlClass monitor is:endl; monitor.display( ); /调用基类成员函数private : /派生类的私有数据Student monitor; /定义子对象(班长)int age;string addr;int main( )Student1 stud1(10010,Wang-li,10001,Li-sun,19,115

37、Beijing Road,Shanghai);stud1.show( ); /输出学生的数据stud1.show_monitor(); /输出子对象的数据return 0;运行时的输出如下:This student is:num: 10010name: Wang-liage: 19address:115 Beijing Road,ShanghaiClass monitor is:num:10001name:Li-sun派生类构造函数的任务应该包括3个部分:(1) 对基类数据成员初始化;(2) 对子对象数据成员初始化;(3) 对派生类数据成员初始化。程序中派生类构造函数首部如下:Student1(int n, string nam,int n1, string nam1,int a, string ad): Student(n,nam),monitor(n1,nam1)在上面的构造函数中有6个形参,前两个作为基类构造函数的参数,第3、第4个作

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 教育专区 > 教案示例

本站为文档C TO C交易模式,本站只提供存储空间、用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。本站仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知淘文阁网,我们立即给予删除!客服QQ:136780468 微信:18945177775 电话:18904686070

工信部备案号:黑ICP备15003705号© 2020-2023 www.taowenge.com 淘文阁