实验8 继承与派生一、实验目的1.理解继承的含义,掌握派生类的定义和实现方法。
2.理解公有继承下基类成员对派生类成员和派生类对象的可见性,能正确地使用继承层次中的各种类成员。
3.理解保护成员在继承中的作用,能够在适当的时候使用保护成员以便派生类成员可以访问基类的部分非公开成员。
4.理解虚基类在类的继承层次中的作用,虚基类的引入对程序运行时的影响,能够对使用虚基类的简单程序写出程序结果。
二、知识要点1.继承继承是C++语言的一种重要机制,它允许在已定义的类的基础上产生新类。
从已定义类产生新类的过程称为派生。
已存在的用来派生新类的类为基类,又称父类。
从已存在的类派生出的新类称为派生类,又称为子类。
如,从哺乳动物类派生出狗类,哺乳动物是父类,狗是子类;从汽车类派生出轿车类,汽车是父类,轿车是子类。
在C++语言中,一个派生类可以从一个基类派生,也可以从多个基类派生。
从一个基类派生的继承称为单继承,从多个基类派生的继承称为多继承。
2.派生类的定义格式(1)单继承的定义格式class<派生类名>:<继承方式><基类名>{<派生类新定义成员>};其中:基类名是已经定义类的名称。
派生类名是新定义的一个类的名字,它是从基类中派生的;派生类是按指定继承方式从基类派生的,继承方式常用的有如下3种:public 表示公有继承private 表示私有继承protected 表示保护继承在单继承中,每个类可以有多个派生类,但是每个派生类只能有一个基类,从而形成树形结构。
(2)多继承的定义格式class<派生类名>:<继承方式1><基类名1>,<继承方式2><基类名2>,……{<派生类新定义成员>};其中继承方式1、继承方式2、……是3种继承方式public、private和protected之一。
多继承与单继承的主要区别从定义格式上看,主要是多继承的基类多于一个。
3.派生类的3种继承方式由下表来理解3种继承方式的各自特点。
基类基类内部函数基类对象private继承方式protected继承方式public继承方式派生类内部函数派生类对象派生类内部函数派生类对象派生类内部函数派生类对象private成员可访问不可访问不可访问不可访问不可访问不可访问不可访问不可访问protected成员可访问不可访问可访问,转为private不可访问可访问,转为protected不可访问可访问,保持protected不可访问public成员可访问可访问可访问,转为private不可访问可访问,转为protected不可访问可访问,保持public可访问4.派生类和基类的关系任何一个类都可以派生出很多个新类,派生类也可以再派生出新类,因此,基类和派生类是相对而言的。
一个基类可以是另一个基类的派生类,这样便形成了复杂的继承结构,出现了类的层次。
一个基类派生出一个派生类,它又做另一个派生类的基类,则原来基类为该派生类的间接基类。
基类和派生类之间的关系可以有以下3种描述。
(1)派生类是基类的具体化类的层次通常反映了客观世界中某种真实的模型。
基类是对若干个派生类的抽象,而派生类是基类的具体化。
基类抽取了它的派生类的公共性,而派生类通过增加行为将抽象类变为某种有用的类型。
(2)派生类是基类定义的延续先定义一个抽象基类,该基类中有些操作并未实现,然后定义非抽象的派生类,实现抽象基类中定义的操作。
例如虚函数就属于此类情况。
这时派生类是抽象的基类的实现,既可以看成是基类定义的延续,这也是派生类的一种常用方法。
(3)派生类是基类的组合。
在多重继承时,一个派生类有多于一个的基类,这时派生类将是所有基类行为的组合。
5.虚基类的引入和说明引进虚基类的真正目的是为了解决二义性的问题。
声明虚基类的方法是:在定义虚基类的直接派生类时,用关键字virtual引出基类名。
6.二义性问题一般来说,在派生类中对基类成员的访问应该是唯一的,但是由于多继承情况下,可能造成对基类中某个成员的访问出现不唯一的情况,则称为对基类成员访问的二义性问题。
由多重继承引起的二义性问题是指:当一个派生类从多个基类派生,而这些基类又有一个共同的基类,则对该基类中说明的成员进行访问时,可能会出现二义性。
7.派生类构造函数和析构函数(1)派生类的对象的数据成员是由基类中说明的数据成员和派生类中说明的数据成员共同构成。
将派生类的对象中由基类说明的数据成员和操作所构成的封装体称为基类子对象,它由基类中的构造函数进行初始化。
(2)构造函数不能够被继承,因此派生类的构造函数必须通过调用基类的构造函数来初始化基类子对象。
所以在定义派生类的构造函数时除了对自己的数据成员进行初始化外,还必须负责调用基类构造函数使基类的数据成员得以初始化,如果派生类中还有子对象时,还应该包含对子对象初始化的构造函数。
(3)派生类构造函数的一般格式如下(派生类名)(<派生类构造函数总参数表>):<基类构造函数>(<参数表1>),<子对象名>(<参数表2>){<派生类中数据成员初始化>}(4)派生类构造函数的调用顺序如下:基类的构造函数-->子对象类的构造函数-->派生类的构造函数(5)当对象被删除时,派生类的析构函数被执行。
由于析构函数不能被继承,因此在执行派生类的析构函数时,基类的析构函数也将被调用。
执行顺序是先执行派生类的析构函数,再执行基类的析构函数,其顺序与执行构造函数时的顺序正好相反。
(6)派生类构造函数使用中应注意的问题:派生类构造函数的定义中可以省略对基类构造函数的调用,其条件是在基类中必须有默认的构造函数或者根本没有定义构造函数。
当然,如果基类中没有定义构造函数,那么派生类根本不必负责调用基类构造函数。
当基类的构造函数使用一个或多个参数时,则派生类必须定义构造函数,提供将参数传递给基类构造函数的途径。
在某些情况下,派生类构造函数的函数体可能为空,仅起到参数传递作用。
三、实验内容和步骤1.定义和使用类的继承关系与定义派生类【实例1】编写一个学生和教师数据输入和显示程序,学生数据有编号、姓名、班级和成绩,教师数据有编号、姓名、职称和部门。
要求将编号、姓名的输入和显示设计成一个类Person,并作为学生数据操作类Student和教师数据操作类Teacher的基类。
题目分析:由题目可以得出需要设计一个Person基类,Teacher类和Student类都是由Person类派生的,即Teacher类和Student类都是由Person类继承而来,并且Teacher类和Student 类都有编号和姓名数据成员,可以把它们作为Person类的公有或保护数据成员。
程序示例:#include<iostream.h>class Person{protected:char name[10];int number;public:void input(){ cin>>name>>number;}void show(){ cout<<name<<"\t"<<number<<endl;}};class Student :public Person{char sclass[10];float score;};class Teacher :public Person{char dept[10];char title[6];};void main(){Student s1;Teacher t1;cout<<"Please input the name and the number of a student:"<<endl;s1.input();s1.show();cout<<"Please input the name and the number of a teacher:"<<endl;t1.input();t1.show();}注意:把input和show两个函数放在Person类中。
实验要求:1)上机运行该程序。
2)为Teacher类编写系别和职称的输入/输出函数;为Student类编写班级和成绩的输入/输出函数。
2.熟悉不同方式下对基类成员的访问控制【实例2】给出下面程序的执行结果。
#include<iostream.h>class A{public:A (int i,int j){ a=i;b=j;}void move(int x,int y){ a+=x;b+=y;}void show(){ cout<<"("<<a<<","<<b<<")"<<endl;}private:int a,b;};class B : public A{public:B(int i,int j,int k,int l) : A(i,j){ x=k;y=l;}void show(){ cout<<x<<","<<y<<endl;}void fun(){ move(3,5);}void f1(){ A::show();}private:int x,y;};void main(){A a(1,2);a.show();B b(3,4,5,6);b.fun();b.show();b.f1();}运行结果:(1,2)(5,6)(6,9)注意:(1)类A和类B中的数据成员都是私有属性,故对它们的访问只能通过成员函数。
(2)注意对象的初始化方法。
实验要求:上机运行程序,并修改已知数据,分析结果。
【实例3】指出下面程序的错误并改正之。
#include<iostream.h>class Point{ int x,y;public:Point (int xx,int yy){ x=xx;y=yy;}void add(int xa,int ya){ x+=xa;y+=ya;}void show(){ cout<<"x="<<x<<","<<"y="<<y<<endl;}};class Rect:private Point{ int len,width;public:Rect (int x,int y,int ll,int ww) : Point (x,y){len=ll;width=ww;}void showRect(){ show();cout<<"length="<<len<<","<<"width="<<width<<endl;}};void main(){ Rect rect(0,2,3,6);rect.add(4,5);rect.showRect();}题目分析:一个类被私有继承之后,其成员在派生类中访问属性会变为private,因而在派生类的对象rect不能直接访问基类的成员函数add。