1第七章继承与派生类的继承与派生类成员的访问控制类型兼容规则单继承与多继承派生类的构造、析构函数类成员的标识与访问深度探索[下半场]2类的继承与派生保持已有类的特性而构造新类的过程称为继承。在已有类的基础上新增自己的特性而产生新类的过程称为派生。被继承的已有类称为基类(或父类)。派生出的新类称为派生类。3继承与派生问题举例工具车轿车面包车小汽车卡车旅行车汽车交通工具类的继承与派生4继承与派生问题举例猴子狮子虎猎豹猫鸟动物类的继承与派生猫科5继承与派生问题举例圆矩形几何形状类的继承与派生6继承与派生问题举例兼职技术人员销售经理管理人员销售人员雇员类的继承与派生7继承与派生的目的继承的目的:实现代码重用。派生的目的:当新的问题出现,原有程序无法解决(或不能完全解决)时,需要对原有程序进行改造。类的继承与派生8派生类的声明class派生类名:继承方式基类名{成员声明;}类的继承与派生9继承方式不同继承方式的影响主要体现在:–派生类成员对基类成员的访问权限–通过派生类对象对基类成员的访问权限三种继承方式–公有继承–私有继承–保护继承类成员的访问控制10公有继承(public)基类的public和protected成员的访问属性在派生类中保持不变,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象只能访问基类的public成员。类成员的访问控制11例7-1公有继承举例//Rectangle.hclassPoint//基类Point类的声明{public://公有函数成员voidInitPoint(floatx=0,floaty=0){this-x=x;this-y=y;}voidMove(floatxOff,floatyOff){x+=xOff;y+=yOff;}floatGetX(){returnx;}floatGetY(){returnx;}private://私有数据成员floatx,y;};//endclassPoint类成员的访问控制classRectangle:publicPoint//派生类声明部分{public://新增公有函数成员voidInitRectangle(floatx,floaty,floatw,floath){InitPoint(x,y);//调用基类公有成员函数this-w=w;this-h=h;}//InitRectanglefloatGetH(){returnh;}floatGetW(){returnw;}private://新增私有数据成员floatw,h;};//endclassRectangle//endofRectangle.h12//7_1.cpp#includeiostream#includecmath#includerectangle.husingnamespacestd;voidmain(void){Rectanglerect;//声明Rectangle类的对象rect.InitRectangle(2,3,20,10);//设置矩形的数据rect.Move(3,2);//移动矩形位置coutThedataofrect(x,y,w,h):endl;coutrect.GetX(),//输出矩形的特征参数rect.GetY(),rect.GetW(),rect.GetH()endl;}//main1314私有继承(private)基类的public和protected成员都以private身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制15例7-2私有继承举例//rectangle.h……//基类Point类的声明classRectangle:privatePoint//派生类声明部分{public://新增公有函数成员voidInitRectangle(floatx,floaty,floatw,floath){InitPoint(x,y);//调用基类公有成员函数this-w=w;this-h=h;}//InitRectanglevoidMove(floatxOff,floatyOff){Point::Move(xOff,yOff);}floatGetX(){returnPoint::GetX();}floatGetY(){returnPoint::GetY();}floatGetH(){returnh;}floatGetW(){returnw;}private://新增私有数据成员floatw,h;};//endclassRectangle类成员的访问控制//7_2.cpp#includeiostream#includecmath#includerectangle.husingnamespacestd;voidmain(void){Rectanglerect;//声明Rectangle类的对象rect.InitRectangle(2,3,20,10);//设置矩形的数据rect.Move(3,2);//移动矩形位置coutThedataofrect(x,y,w,h):endl;coutrect.GetX(),//输出矩形的特征参数rect.GetY(),rect.GetW(),rect.GetH()endl;}//main1617保护继承(protected)基类的public和protected成员都以protected身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员类成员的访问控制18protected成员的特点与作用对建立其所在类对象的模块来说,它与private成员的性质相同。对于其派生类来说,它与public成员的性质相同。既实现了数据隐藏,又方便继承,实现代码重用。类成员的访问控制19protected成员举例classA{protected:intx;};voidmain(void){Aa;a.x=5;//错误}//main类成员的访问控制classA{protected:intx;};classB:publicA{public:voidfunc();};voidB::func(){x=5;//正确}20类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象,反之则禁止。具体表现在:–派生类的对象可以隐含转换为基类对象。–派生类的对象可以初始化基类的引用。–派生类的指针可以隐含转换为基类的指针。通过基类对象名、指针只能使用从基类继承的成员类型兼容21例7-3类型兼容规则举例#includeiostreamusingnamespacestd;classBase1{//基类Base1定义public:voiddisplay()const{coutBase1::display()endl;}};//endclassBase1类型兼容classBase2:publicBase1{//公有派生类Base2定义public:voiddisplay()const{coutBase2::display()endl;}};//endclassBase2classDerived:publicBase2{//公有派生类Derived定义public:voiddisplay()const{coutDerived::display()endl;}};//endclassDerivedvoidfun(Base1*ptr){//参数为指向基类对象的指针ptr-display();//对象指针-成员名}//fun22voidmain(void){//主函数Base1base1;//声明Base1类对象Base2base2;//声明Base2类对象Derivedderived;//声明Derived类对象//用Base1对象的指针调用fun函数fun(&base1);//用Base2对象的指针调用fun函数fun(&base2);//用Derived对象的指针调用fun函数fun(&derived);}//main运行结果:Base1::display()Base1::display()Base1::display()2324基类与派生类的对应关系单继承–派生类只从一个基类派生。多继承–派生类从多个基类派生。多重派生–由一个基类派生出多个不同的派生类。多层派生–派生类又作为基类,继续派生新的类。单继承与多继承25多继承时派生类的声明class派生类名:继承方式1基类名1,继承方式2基类名2,...{成员声明;}注意:每一个“继承方式”,只用于限制紧随其后之基类的继承。单继承与多继承26多继承举例classA{public:voidsetA(int);voidshowA()const;private:inta;};//endclassAclassB{public:voidsetB(int);voidshowB()const;private:intb;};//endclassBclassC:publicA,privateB{public:voidsetC(int,int,int);voidshowC()const;privateconst:intc;};//endclassC单继承与多继承voidA::setA(intx){a=x;}//setAvoidB::setB(intx){b=x;}//setBvoidC::setC(intx,inty,intz){//派生类成员直接访问基类的公有成员setA(x);setB(y);c=z;}//setC//其他函数实现略//主函数voidmain(void){Cobj;obj.setA(5);obj.showA();obj.setC(6,7,9);obj.showC();//obj.setB(6);错误//obj.showB();错误}//main2728继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己的构造函数。定义构造函数时,只需要对本类中新增成员进行初始化,对继承来的基类成员的初始化,自动调用基类构造函数完成。派生类的构造函数需要给基类的构造函数传递参数派生类的构造、析构函数29单一继承时的构造函数派生类名::派生类名(基类所需的形参,本类成员所需的形参):基类名(参数表){本类成员初始化赋值语句;};派生类的构造、析构函数30单一继承时的构造函数举例#includeiostreamusingnamespacestd;classB{public:B();B(inti);~B();voidprint()const;private:intb;};//classB派生类的构造、析构函数B::B(){b=0;coutB'sdefaultconstructorcalled.endl;}//B()B::B(inti){b=i;coutB'sconstructorcalled.endl;}//B(int)B::~B(){coutB'sdestructorcalled.endl;}//~B()voidB::print()const{coutbendl;}//print31classC:publicB{public:C();C(inti,intj);~C();voidprint()const;private:intc;};//