版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
(最新整理)C++语言程序设计(清华大学郑莉)七2021/7/261(最新整理)C++语言程序设计(清华大学郑莉)七2021/7第七章继承与派生清华大学郑莉C++语言程序设计2021/7/262第七章继承与派生清华大学郑莉C++语言程序设计本章主要内容类的继承类成员的访问控制单继承与多继承派生类的构造、析构函数类成员的标识与访问深度探索2021/7/263本章主要内容类的继承2021/7/263类的继承与派生保持已有类的特性而构造新类的过程称为继承。在已有类的基础上新增自己的特性而产生新类的过程称为派生。被继承的已有类称为基类(或父类)。派生出的新类称为派生类。2021/7/264类的继承与派生保持已有类的特性而构造新类的过程称为继承。20继承与派生问题举例类的继承与派生2021/7/265继承与派生问题举例类的继承与派生2021/7/265继承与派生问题举例类的继承与派生猫科2021/7/266继承与派生问题举例类的继承与派生猫科2021/7/266继承与派生问题举例类的继承与派生2021/7/267继承与派生问题举例类的继承与派生2021/7/267继承与派生问题举例类的继承与派生2021/7/268继承与派生问题举例类的继承与派生2021/7/268继承与派生的目的继承的目的:实现代码重用。派生的目的:当新的问题出现,原有程序无法解决(或不能完全解决)时,需要对原有程序进行改造。类的继承与派生2021/7/269继承与派生的目的继承的目的:实现代码重用。类的继承与派生20派生类的声明class派生类名:继承方式基类名{
成员声明;}类的继承与派生2021/7/2610派生类的声明class派生类名:继承方式基类名类的继承继承方式不同继承方式的影响主要体现在:派生类成员对基类成员的访问权限通过派生类对象对基类成员的访问权限三种继承方式公有继承私有继承保护继承类成员的访问控制2021/7/2611继承方式不同继承方式的影响主要体现在:类成员的访问控制202公有继承(public)基类的public和protected成员的访问属性在派生类中保持不变,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象只能访问基类的public成员。类成员的访问控制2021/7/2612公有继承(public)基类的public和protecte例7-1公有继承举例classPoint{ //基类Point类的定义public: //公有函数成员
voidinitPoint(floatx=0,floaty=0){this->x=x;this->y=y;} voidmove(floatoffX,floatoffY){x+=offX;y+=offY;} floatgetX()const{returnx;} floatgetY()const{returny;}private: //私有数据成员
floatx,y;};类成员的访问控制2021/7/2613例7-1公有继承举例classPoint{ //基类PclassRectangle:publicPoint{ //派生类定义部分public: //新增公有函数成员
voidinitRectangle(floatx,floaty,floatw,floath){
initPoint(x,y);//调用基类公有成员函数
this->w=w; this->h=h; } floatgetH()const{returnh;} floatgetW()const{returnw;}private: //新增私有数据成员
floatw,h;};142021/7/2614classRectangle:publicPoint#include<iostream>#include<cmath>usingnamespacestd;intmain(){ Rectanglerect; //定义Rectangle类的对象 //设置矩形的数据
rect.initRectangle(2,3,20,10);
rect.move(3,2); //移动矩形位置
cout<<"Thedataofrect(x,y,w,h):"<<endl; //输出矩形的特征参数 cout<<rect.getX()<<","
<<rect.getY()<<"," <<rect.getW()<<"," <<rect.getH()<<endl; return0;}152021/7/2615#include<iostream>152021/7/26私有继承(private)基类的public和protected成员都以private身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制2021/7/2616私有继承(private)基类的public和protect例7-2私有继承举例classRectangle:privatePoint{ //派生类定义部分public: //新增公有函数成员
voidinitRectangle(floatx,floaty,floatw,floath){
initPoint(x,y);//调用基类公有成员函数
this->w=w; this->h=h; } voidmove(floatoffX,floatoffY){
Point::move(offX,offY); } floatgetX()const{returnPoint::getX();} floatgetY()const{returnPoint::getY();} floatgetH()const{returnh;} floatgetW()const{returnw;}private: //新增私有数据成员
floatw,h;};类成员的访问控制2021/7/2617例7-2私有继承举例classRectangle:pr#include<iostream>#include<cmath>usingnamespacestd;intmain(){
Rectanglerect; //定义Rectangle类的对象
rect.initRectangle(2,3,20,10); //设置矩形的数据
rect.move(3,2); //移动矩形位置
cout<<"Thedataofrect(x,y,w,h):"<<endl; cout<<rect.getX()<<"," //输出矩形的特征参数
<<rect.getY()<<"," <<rect.getW()<<"," <<rect.getH()<<endl; return0;}182021/7/2618#include<iostream>182021/7/26保护继承(protected)基类的public和protected成员都以protected身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员类成员的访问控制2021/7/2619保护继承(protected)基类的public和proteprotected成员的特点与作用对建立其所在类对象的模块来说,它与private成员的性质相同。对于其派生类来说,它与public成员的性质相同。既实现了数据隐藏,又方便继承,实现代码重用。类成员的访问控制2021/7/2620protected成员的特点与作用对建立其所在类对象的模块例7-3protected成员举例classA{protected: intx;};intmain(){ Aa; a.x=5;//错误}类成员的访问控制2021/7/2621例7-3protected成员举例classA{类成classA{protected: intx;};classB:publicA{public: voidfunction();};voidB:function(){ x=5;//正确}222021/7/2622classA{222021/7/2622类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象,反之则禁止。具体表现在:派生类的对象可以隐含转换为基类对象。派生类的对象可以初始化基类的引用。派生类的指针可以隐含转换为基类的指针。通过基类对象名、指针只能使用从基类继承的成员类型兼容2021/7/2623类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象例7-4类型兼容规则举例#include<iostream>usingnamespacestd;classBase1{//基类Base1定义public: voiddisplay()const{ cout<<"Base1::display()"<<endl; }};类型兼容2021/7/2624例7-4类型兼容规则举例#include<iostreclassBase2:publicBase1{//公有派生类Base2定义public: voiddisplay()const{ cout<<"Base2::display()"<<endl; }};classDerived:publicBase2{//公有派生类Derived定义public: voiddisplay()const{ cout<<"Derived::display()"<<endl; }};voidfun(Base1*ptr){//参数为指向基类对象的指针
ptr->display(); //"对象指针->成员名"}252021/7/2625classBase2:publicBase1{//intmain(){ //主函数
Base1base1; //声明Base1类对象
Base2base2; //声明Base2类对象
Derivedderived; //声明Derived类对象
//用Base1对象的指针调用fun函数 fun(&base1); //用Base2对象的指针调用fun函数
fun(&base2); //用Derived对象的指针调用fun函数fun(&derived);
return0;}运行结果:B0::display()B0::display()B0::display()262021/7/2626intmain(){ //主函数运行结果:262021/基类与派生类的对应关系单继承派生类只从一个基类派生。多继承派生类从多个基类派生。多重派生由一个基类派生出多个不同的派生类。多层派生派生类又作为基类,继续派生新的类。单继承与多继承2021/7/2627基类与派生类的对应关系单继承单继承与多继承2021/7/26多继承时派生类的声明class派生类名:继承方式1基类名1,
继承方式2基类名2,...{
成员声明;}注意:每一个“继承方式”,只用于限制对紧随其后之基类的继承。单继承与多继承2021/7/2628多继承时派生类的声明class派生类名:继承方式1基类多继承举例classA{public: voidsetA(int); voidshowA()const;private: inta;};classB{public: voidsetB(int); voidshowB()const;private: intb;};classC:publicA,privateB{public: voidsetC(int,int,int); voidshowC()const;privateconst: intc;};单继承与多继承2021/7/2629多继承举例classA{private:单继承与多继承2voidA::setA(intx){ a=x;}voidB::setB(intx){ b=x;}voidC::setC(intx,inty,intz){ //派生类成员直接访问基类的 //公有成员 setA(x); setB(y); c=z;}//其他函数实现略intmain(){ Cobj; obj.setA(5); obj.showA(); obj.setC(6,7,9); obj.showC();//obj.setB(6);错误//obj.showB();错误 return0;}302021/7/2630voidA::setA(intx){intmain继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己的构造函数。定义构造函数时,只需要对本类中新增成员进行初始化,对继承来的基类成员的初始化,自动调用基类构造函数完成。派生类的构造函数需要给基类的构造函数传递参数派生类的构造、析构函数2021/7/2631继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己单一继承时的构造函数派生类名::派生类名(基类所需的形参,本类成员所需的形参):基类名(参数表){
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/2632单一继承时的构造函数派生类名::派生类名(基类所需的形参,本单一继承时的构造函数举例#include<iostream>usingnamecpacestd;classB{public: B(); B(inti); ~B(); voidprint()const;private: intb;};派生类的构造、析构函数2021/7/2633单一继承时的构造函数举例#include<iostream>B::B(){ b=0; cout<<"B'sdefaultconstructorcalled."<<endl;}B::B(inti){ b=i; cout<<"B'sconstructorcalled."<<endl;}B::~B(){ cout<<"B'sdestructorcalled."<<endl;}voidB::print()const{ cout<<b<<endl;}342021/7/2634B::B(){342021/7/2634classC:publicB{public: C(); C(inti,intj); ~C(); voidprint()const;private: intc;};352021/7/2635classC:publicB{352021/7/26C::C(){ c=0; cout<<"C'sdefaultconstructorcalled."<<endl;}C::C(inti,intj):B(i){ c=j; cout<<"C'sconstructorcalled."<<endl;}C::~C(){ cout<<"C'sdestructorcalled."<<endl;}voidC::print()const{ B::print(); cout<<c<<endl;}intmain(){ Cobj(5,6); obj.print(); return0;}362021/7/2636C::C(){362021/7/2636多继承时的构造函数派生类名::派生类名(参数表):基类名1(基类1初始化参数表),基类名2(基类2初始化参数表),...基类名n(基类n初始化参数表){
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/2637多继承时的构造函数派生类名::派生类名(参数表):基类名1(派生类与基类的构造函数当基类中声明有缺省构造函数或未声明构造函数时,派生类构造函数可以不向基类构造函数传递参数,也可以不声明,构造派生类的对象时,基类的缺省构造函数将被调用。当需要执行基类中带形参的构造函数来初始化基类数据时,派生类构造函数应在初始化列表中为基类构造函数提供参数。派生类的构造、析构函数2021/7/2638派生类与基类的构造函数当基类中声明有缺省构造函数或未声明构造多继承且有内嵌对象时
的构造函数派生类名::派生类名(形参表):基类名1(参数),基类名2(参数),...基类名n(参数),新增成员对象的初始化{
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/2639多继承且有内嵌对象时
的构造函数派生类名::派生类名(形参表构造函数的执行顺序1.调用基类构造函数,调用顺序按照它们被继承时声明的顺序(从左向右)。2.对成员对象进行初始化,初始化顺序按照它们在类中声明的顺序。3.执行派生类的构造函数体中的内容。派生类的构造、析构函数2021/7/2640构造函数的执行顺序1.调用基类构造函数,调用顺序按照它们被拷贝构造函数若建立派生类对象时没有编写拷贝构造函数,编译器会生成一个隐含的拷贝构造函数,该函数先调用基类的拷贝构造函数,再为派生类新增的成员对象执行拷贝。若编写派生类的拷贝构造函数,则需要为基类相应的拷贝构造函数传递参数。例如:C::C(C&c1):B(c1){…}派生类的构造、析构函数2021/7/2641拷贝构造函数若建立派生类对象时没有编写拷贝构造函数,编译器会例7-4派生类构造函数举例#include<iostream>usingnamespacestd;classBase1{ //基类Base1,构造函数有参数public: Base1(inti){cout<<"ConstructingBase1"<<i<<endl;}};classBase2{ //基类Base2,构造函数有参数public: Base2(intj){cout<<"ConstructingBase2"<<j<<endl;}};classBase3{ //基类Base3,构造函数无参数public: Base3(){cout<<"ConstructingBase3*"<<endl;}};派生类的构造、析构函数2021/7/2642例7-4派生类构造函数举例#include<iostreclassDerived:publicBase2,publicBase1,publicBase3{//派生新类Derived,注意基类名的顺序public: //派生类的公有成员
Derived(inta,intb,intc,intd):Base1(a),member2(d),member1(c),Base2(b) {} //注意基类名的个数与顺序,//注意成员对象名的个数与顺序private: //派生类的私有成员对象
Base1member1; Base2member2; Base3member3;};intmain(){ Derivedobj(1,2,3,4); return0;}运行结果:constructingBase22constructingBase11constructingBase3*constructingBase13constructingBase24constructingBase3*432021/7/2643classDerived:publicBase2,p继承时的析构函数析构函数也不被继承,派生类自行声明声明方法与一般(无继承关系时)类的析构函数相同。不需要显式地调用基类的析构函数,系统会自动隐式调用。析构函数的调用次序与构造函数相反。派生类的构造、析构函数2021/7/2644继承时的析构函数析构函数也不被继承,派生类自行声明派生类的构例7-5派生类析构函数举例派生类的构造、析构函数#include<iostream>usingnamespacestd;classBase1{ //基类Base1,构造函数有参数public: Base1(inti){cout<<"ConstructingBase1"<<i<<endl;} ~Base1(){cout<<"DestructingBase1"<<endl;}};classBase2{ //基类Base2,构造函数有参数public: Base2(intj){cout<<"ConstructingBase2"<<j<<endl;} ~Base2(){cout<<"DestructingBase2"<<endl;}};classBase3{ //基类Base3,构造函数无参数public: Base3(){cout<<"ConstructingBase3*"<<endl;} ~Base3(){cout<<"DestructingBase3"<<endl;}};2021/7/2645例7-5派生类析构函数举例派生类的构造、析构函数#incclassDerived:publicBase2,publicBase1,publicBase3{//派生新类Derived,注意基类名的顺序public: //派生类的公有成员
Derived(inta,intb,intc,intd):Base1(a),member2(d),member1(c),Base2(b){} //注意基类名的个数与顺序,注意成员对象名的个数与顺序private: //派生类的私有成员对象
Base1member1; Base2member2; Base3member3;};intmain(){ Derivedobj(1,2,3,4); return0;}462021/7/2646classDerived:publicBase2,p例7-5运行结果ConstructingBase22ConstructingBase11ConstructingBase3*ConstructingBase13ConstructingBase24ConstructingBase3*DestructingBase3DestructingBase2DestructingBase1DestructingBase3DestructingBase1DestructingBase22021/7/2647例7-5运行结果ConstructingBase222同名隐藏规则当派生类与基类中有相同成员时:若未强行指名,则通过派生类对象使用的是派生类中的同名成员。如要通过派生类对象访问基类中被隐藏的同名成员,应使用基类名限定。派生类成员的标识与访问2021/7/2648同名隐藏规则当派生类与基类中有相同成员时:派生类成员的标识与例7-6多继承同名隐藏举例(1)派生类成员的标识与访问#include<iostream>usingnamespacestd;classBase1{ //定义基类Base1public: intvar; voidfun(){cout<<"MemberofBase1"<<endl;}};classBase2{ //定义基类Base2public: intvar; voidfun(){cout<<"MemberofBase2"<<endl;}};classDerived:publicBase1,publicBase2{//定义派生类Derivedpublic: intvar; //同名数据成员
voidfun(){cout<<"MemberofDerived"<<endl;} //同名函数成员};2021/7/2649例7-6多继承同名隐藏举例(1)派生类成员的标识与访问#intmain(){ Derivedd; Derived*p=&d; d.var=1; //对象名.成员名标识
d.fun(); //访问Derived类成员
d.Base1::var=2; //作用域分辨符标识
d.Base1::fun(); //访问Base1基类成员
p->Base2::var=3; //作用域分辨符标识
p->Base2::fun(); //访问Base2基类成员
return0;}502021/7/2650intmain(){502021/7/2650二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成员时,将出现访问时的二义性(不确定性)——采用虚函数(参见第8章)或同名隐藏规则来解决。当派生类从多个基类派生,而这些基类又从同一个基类派生,则在访问此共同基类中的成员时,将产生二义性——采用虚基类来解决。派生类成员的标识与访问2021/7/2651二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成二义性问题举例(一)classA{public: voidf();};classB{public: voidf(); voidg()};classC:publicA,piblicB{public: voidg(); voidh();};如果定义:Cc1;则c1.f()具有二义性而c1.g()无二义性(同名隐藏)派生类成员的标识与访问2021/7/2652二义性问题举例(一)classA{classC:pu二义性的解决方法解决方法一:用类名来限定
c1.A::f()或c1.B::f()解决方法二:同名隐藏
在C中声明一个同名成员函数f(),f()再根据需要调用A::f()或B::f()派生类成员的标识与访问2021/7/2653二义性的解决方法解决方法一:用类名来限定
c1.A::f()二义性问题举例(二)classB{public: intb;}classB1:publicB{private: intb1;};classB2:publicB{private: intb2;};classC:publicB1,publicB2{public: intf();private: intd;}派生类成员的标识与访问有二义性:Cc;c.bc.B::b无二义性:c.B1::bc.B2::b2021/7/2654二义性问题举例(二)classB{classC:p虚基类虚基类的引入用于有共同基类的场合声明以virtual修饰说明基类
例:classB1:virtualpublicB作用主要用来解决多继承时可能发生的对同一基类继承多次而产生的二义性问题.为最远的派生类提供唯一的基类成员,而不重复产生多次拷贝注意:在第一级继承时就要将共同基类设计为虚基类。2021/7/2655虚基类虚基类的引入2021/7/2655虚基类举例classB{public:intb;};classB1:virtualpublicB{public:intb1;};classB2:virtualpublicB{public:intb2;};classC:publicB1,publicB2{public:floatd;};下面的访问是正确的:Ccobj;cobj.b;
虚基类2021/7/2656虚基类举例classB{public:intb;例7-8虚基类举例
虚基类DerivedBase0::var0:intBase1::var1:intBase2::var2:intvar:intBase0::fun0():voidfun():voidBase1+var1:intBase2+var2:intDerived+var:int+fun():void<<virtual>>Base0+var0:int+fun0():void2021/7/2657例7-8虚基类举例虚基类Derive#include<iostream>usingnamespacestd;classBase0{ //定义基类Base0public: intvar0; voidfun0(){cout<<"MemberofBase0"<<endl;}};classBase1:virtualpublicBase0{ //定义派生类Base1public: //新增外部接口
intvar1;};classBase2:virtualpublicBase0{ //定义派生类Base2public: //新增外部接口
intvar2;};582021/7/2658#include<iostream>582021/7/26classDerived:publicBase1,publicBase2{ //定义派生类Derivedpublic: //新增外部接口
intvar; voidfun(){ cout<<"MemberofDerived"<<endl; }};intmain(){ //程序主函数
Derivedd; //定义Derived类对象d d.var0=2; //直接访问虚基类的数据成员
d.fun0(); //直接访问虚基类的函数成员
return0;}592021/7/2659classDerived:publicBase1,p虚基类及其派生类构造函数建立对象时所指定的类称为最(远)派生类。虚基类的成员是由最派生类的构造函数通过调用虚基类的构造函数进行初始化的。在整个继承结构中,直接或间接继承虚基类的所有派生类,都必须在构造函数的成员初始化表中给出对虚基类的构造函数的调用。如果未列出,则表示调用该虚基类的默认构造函数。在建立对象时,只有最派生类的构造函数调用虚基类的构造函数,该派生类的其他基类对虚基类构造函数的调用被忽略。
虚基类2021/7/2660虚基类及其派生类构造函数建立对象时所指定的类称为最(远)派生有虚基类时的构造函数举例
虚基类#include<iostream>usingnamespacestd;classBase0{ //定义基类Base0public: Base0(intvar):var0(var){} intvar0; voidfun0(){cout<<"MemberofBase0"<<endl;}};classBase1:virtualpublicBase0{ //定义派生类Base1public: //新增外部接口
Base1(intvar):Base0(var){} intvar1;};classBase2:virtualpublicBase0{ //定义派生类Base2public: //新增外部接口
Base2(intvar):Base0(var){} intvar2;};2021/7/2661有虚基类时的构造函数举例虚基类#inclassDerived:publicBase1,publicBase2{ //定义派生类Derivedpublic: //新增外部接口
Derived(intvar):Base0(var),Base1(var),Base2(var){} intvar; voidfun(){cout<<"MemberofDerived"<<endl;}};intmain(){ //程序主函数
Derivedd(1); //定义Derived类对象d d.var0=2; //直接访问虚基类的数据成员
d.fun0(); //直接访问虚基类的函数成员
return0;}622021/7/2662classDerived:publicBase1,p组合与继承组合与继承:通过已有类来构造新类的两种基本方式组合:B类中存在一个A类型的内嵌对象有一个(has-a)关系:表明每个B类型对象“有一个”A类型对象A类型对象与B类型对象是部分与整体关系B类型的接口不会直接作为A类型的接口深度探索2021/7/2663组合与继承组合与继承:通过已有类来构造新类的两种基本方式深“has-a”举例classEngine{ //发动机类public: voidwork(); //发动机运转
……};classWheel{ //轮子类public: voidroll(); //轮子转动
……};classAutomobile{ //汽车类public: voidmove(); //汽车移动private: Engineengine; //汽车引擎
Wheelwheels[4];//4个车轮
……};意义一辆汽车有一个发动机一辆汽车有四个轮子接口作为整体的汽车不再具备发动机的运转功能,和轮子的转动功能,但通过将这些功能的整合,具有了自己的功能——移动深度探索2021/7/2664“has-a”举例classEngine{ //发动机类公有继承的意义公有继承:A类是B类的公有基类是一个(is-a)关系:表明每个B类型对象“是一个”A类型对象A类型对象与B类型对象是一般与特殊关系回顾类的兼容性原则:在需要基类对象的任何地方,都可以使用公有派生类的对象来替代B类型对象包括A类型的全部接口深度探索2021/7/2665公有继承的意义公有继承:A类是B类的公有基类深度探索2“is-a”举例classTruck:publicAutomobile{//卡车public: voidload(…); //装货
voiddump(…); //卸货private: ……};classPumper:publicAutomobile{//消防车public: voidwater(); //喷水private: ……};意义卡车是汽车消防车是汽车接口卡车和消防车具有汽车的通用功能(移动)它们还各自具有自己的功能(卡车:装货、卸货;消防车:喷水)深度探索2021/7/2666“is-a”举例classTruck:publicAu派生类对象的内存布局派生类对象的内存布局因编译器而异内存布局应使类型兼容规则便于实现一个基类指针,无论其指向基类对象,还是派生类对象,通过它来访问一个基类中定义的数据成员,都可以用相同的步骤不同情况下的内存布局单继承:基类数据在前,派生类新增数据在后多继承:各基类数据按顺序在前,派生类新增数据在后虚继承:需要增加指针,间接访虚基类数据深度探索2021/7/2667派生类对象的内存布局派生类对象的内存布局深度探索202单继承情形classBase{…};classDerived:publicBase{…};Derived*pd=newDerived();Base*pb=pd;Derived类型指针pd转换为Base类型指针时,地址不需要改变Base类数据成员Derived类新增数据成员pb,pdDerived对象深度探索2021/7/2668单继承情形classBase{…};Derived类多继承情形classBase1{…};classBase2{…};classDerived:publicBase1,publicBase2{…};Derived*pd=newDerived();Base1*pb1=pd;Base2*pb2=pd;Derived类型指针pd转换为Base2类型指针时,原地址需要增加一个偏移量Base1类数据成员Base2类数据成员pb1,pdDerived对象pb2Derived类新增数据成员深度探索2021/7/2669多继承情形classBase1{…};Derived虚拟继承情形classBase0{…};classBase1:virtualpublicBase0{…};classBase2:virtualpublicBase0{…};classDerived:publicBase1,publicBase2{…};Derived*pd=newDerived();Base1*pb1=pd;Base2*pb2=pd;Base0*pb0=pb1;Base0类
数据成员Base1类新增数据成员pb1,pdBase0指针Base1类新增数据成员Base0指针Derived对象pb2pb0通过指针间接访问虚基类的数据成员深度探索2021/7/2670虚拟继承情形classBase0{…};Base0类基类向派生的转换基类向派生类的转换基类指针可以转换为派生类指针基类引用可以转换为派生类引用需要用static_cast显式转换例:Base*pb=newDerived();Derived*pd=static_cast<Derived*>(pd);Derivedd;Base&rb=d;Derived&rb=static_cast<Derived&>(rb);深度探索2021/7/2671基类向派生的转换基类向派生类的转换深度探索2021/7类型转换时的注意事项(1)基类对象一般无法被显式转换为派生类对象对象到对象的转换,需要调用构造函数创建新的对象派生类的拷贝构造函数无法接受基类对象作为参数执行基类向派生类的转换时,一定要确保被转换的指针和引用所指向或引用的对象符合转换的目的类型:对于Derived*pd=static_cast<Derived*>(pb);一定要保证pb所指向的对象具有Derived类型,或者是Derived类型的派生类。深度探索2021/7/2672类型转换时的注意事项(1)基类对象一般无法被显式转换为派生类类型转换时的注意事项(2)如果A类型是B类型的虚拟基类,A类型指针无法通过static_cast隐含转换为B类型的指针可以结合虚继承情况下的对象内存布局,思考为什么不允许这种转换void指针参加的转换,可能导致不可预期的后果:例:(Base2是Derived的第二个公共基类)Derived*pd=newDerived();void*pv=pd; //将Derived指针转换为void指针Base2*pb=static_cast<Base2*>(pv);转换后pb与pd有相同的地址,而正常的转换下应有一个偏移量结论:有void指针参与的转换,兼容性规则不适用更安全更灵活的基类向派生类转换方式——dynamic_cast,将在下一讲介绍深度探索2021/7/2673类型转换时的注意事项(2)如果A类型是B类型的虚拟基类,A类小结与复习建议主要内容类的继承、类成员的访问控制、单继承与多继承、派生类的构造和析构函数、类成员的标识与访问达到的目标理解类的继承关系,学会使用继承关系实现代码的重用。实验任务实验七2021/7/2674小结与复习建议主要内容2021/7/26742021/7/26752021/7/2675(最新整理)C++语言程序设计(清华大学郑莉)七2021/7/2676(最新整理)C++语言程序设计(清华大学郑莉)七2021/7第七章继承与派生清华大学郑莉C++语言程序设计2021/7/2677第七章继承与派生清华大学郑莉C++语言程序设计本章主要内容类的继承类成员的访问控制单继承与多继承派生类的构造、析构函数类成员的标识与访问深度探索2021/7/2678本章主要内容类的继承2021/7/263类的继承与派生保持已有类的特性而构造新类的过程称为继承。在已有类的基础上新增自己的特性而产生新类的过程称为派生。被继承的已有类称为基类(或父类)。派生出的新类称为派生类。2021/7/2679类的继承与派生保持已有类的特性而构造新类的过程称为继承。20继承与派生问题举例类的继承与派生2021/7/2680继承与派生问题举例类的继承与派生2021/7/265继承与派生问题举例类的继承与派生猫科2021/7/2681继承与派生问题举例类的继承与派生猫科2021/7/266继承与派生问题举例类的继承与派生2021/7/2682继承与派生问题举例类的继承与派生2021/7/267继承与派生问题举例类的继承与派生2021/7/2683继承与派生问题举例类的继承与派生2021/7/268继承与派生的目的继承的目的:实现代码重用。派生的目的:当新的问题出现,原有程序无法解决(或不能完全解决)时,需要对原有程序进行改造。类的继承与派生2021/7/2684继承与派生的目的继承的目的:实现代码重用。类的继承与派生20派生类的声明class派生类名:继承方式基类名{
成员声明;}类的继承与派生2021/7/2685派生类的声明class派生类名:继承方式基类名类的继承继承方式不同继承方式的影响主要体现在:派生类成员对基类成员的访问权限通过派生类对象对基类成员的访问权限三种继承方式公有继承私有继承保护继承类成员的访问控制2021/7/2686继承方式不同继承方式的影响主要体现在:类成员的访问控制202公有继承(public)基类的public和protected成员的访问属性在派生类中保持不变,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象只能访问基类的public成员。类成员的访问控制2021/7/2687公有继承(public)基类的public和protecte例7-1公有继承举例classPoint{ //基类Point类的定义public: //公有函数成员
voidinitPoint(floatx=0,floaty=0){this->x=x;this->y=y;} voidmove(floatoffX,floatoffY){x+=offX;y+=offY;} floatgetX()const{returnx;} floatgetY()const{returny;}private: //私有数据成员
floatx,y;};类成员的访问控制2021/7/2688例7-1公有继承举例classPoint{ //基类PclassRectangle:publicPoint{ //派生类定义部分public: //新增公有函数成员
voidinitRectangle(floatx,floaty,floatw,floath){
initPoint(x,y);//调用基类公有成员函数
this->w=w; this->h=h; } floatgetH()const{returnh;} floatgetW()const{returnw;}private: //新增私有数据成员
floatw,h;};892021/7/2689classRectangle:publicPoint#include<iostream>#include<cmath>usingnamespacestd;intmain(){ Rectanglerect; //定义Rectangle类的对象 //设置矩形的数据
rect.initRectangle(2,3,20,10);
rect.move(3,2); //移动矩形位置
cout<<"Thedataofrect(x,y,w,h):"<<endl; //输出矩形的特征参数 cout<<rect.getX()<<","
<<rect.getY()<<"," <<rect.getW()<<"," <<rect.getH()<<endl; return0;}902021/7/2690#include<iostream>152021/7/26私有继承(private)基类的public和protected成员都以private身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制2021/7/2691私有继承(private)基类的public和protect例7-2私有继承举例classRectangle:privatePoint{ //派生类定义部分public: //新增公有函数成员
voidinitRectangle(floatx,floaty,floatw,floath){
initPoint(x,y);//调用基类公有成员函数
this->w=w; this->h=h; } voidmove(floatoffX,floatoffY){
Point::move(offX,offY); } floatgetX()const{returnPoint::getX();} floatgetY()const{returnPoint::getY();} floatgetH()const{returnh;} floatgetW()const{returnw;}private: //新增私有数据成员
floatw,h;};类成员的访问控制2021/7/2692例7-2私有继承举例classRectangle:pr#include<iostream>#include<cmath>usingnamespacestd;intmain(){
Rectanglerect; //定义Rectangle类的对象
rect.initRectangle(2,3,20,10); //设置矩形的数据
rect.move(3,2); //移动矩形位置
cout<<"Thedataofrect(x,y,w,h):"<<endl; cout<<rect.getX()<<"," //输出矩形的特征参数
<<rect.getY()<<"," <<rect.getW()<<"," <<rect.getH()<<endl; return0;}932021/7/2693#include<iostream>182021/7/26保护继承(protected)基类的public和protected成员都以protected身份出现在派生类中,但基类的private成员不可直接访问。派生类中的成员函数可以直接访问基类中的public和protected成员,但不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员类成员的访问控制2021/7/2694保护继承(protected)基类的public和proteprotected成员的特点与作用对建立其所在类对象的模块来说,它与private成员的性质相同。对于其派生类来说,它与public成员的性质相同。既实现了数据隐藏,又方便继承,实现代码重用。类成员的访问控制2021/7/2695protected成员的特点与作用对建立其所在类对象的模块例7-3protected成员举例classA{protected: intx;};intmain(){ Aa; a.x=5;//错误}类成员的访问控制2021/7/2696例7-3protected成员举例classA{类成classA{protected: intx;};classB:publicA{public: voidfunction();};voidB:function(){ x=5;//正确}972021/7/2697classA{222021/7/2622类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象,反之则禁止。具体表现在:派生类的对象可以隐含转换为基类对象。派生类的对象可以初始化基类的引用。派生类的指针可以隐含转换为基类的指针。通过基类对象名、指针只能使用从基类继承的成员类型兼容2021/7/2698类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象例7-4类型兼容规则举例#include<iostream>usingnamespacestd;classBase1{//基类Base1定义public: voiddisplay()const{ cout<<"Base1::display()"<<endl; }};类型兼容2021/7/2699例7-4类型兼容规则举例#include<iostreclassBase2:publicBase1{//公有派生类Base2定义public: voiddisplay()const{ cout<<"Base2::display()"<<endl; }};classDerived:publicBase2{//公有派生类Derived定义public: voiddisplay()const{ cout<<"Derived::display()"<<endl; }};voidfun(Base1*ptr){//参数为指向基类对象的指针
ptr->display(); //"对象指针->成员名"}1002021/7/26100classBase2:publicBase1{//intmain(){ //主函数
Base1base1; //声明Base1类对象
Base2base2; //声明Base2类对象
Derivedderived; //声明Derived类对象
//用Base1对象的指针调用fun函数 fun(&base1); //用Base2对象的指针调用fun函数
fun(&base2); //用Derived对象的指针调用fun函数fun(&derived);
return0;}运行结果:B0::display()B0::display()B0::display()1012021/7/26101intmain(){ //主函数运行结果:262021/基类与派生类的对应关系单继承派生类只从一个基类派生。多继承派生类从多个基类派生。多重派生由一个基类派生出多个不同的派生类。多层派生派生类又作为基类,继续派生新的类。单继承与多继承2021/7/26102基类与派生类的对应关系单继承单继承与多继承2021/7/26多继承时派生类的声明class派生类名:继承方式1基类名1,
继承方式2基类名2,...{
成员声明;}注意:每一个“继承方式”,只用于限制对紧随其后之基类的继承。单继承与多继承2021/7/26103多继承时派生类的声明class派生类名:继承方式1基类多继承举例classA{public: voidsetA(int); voidshowA()const;private: inta;};classB{public: voidsetB(int); voidshowB()const;private: intb;};classC:publicA,privateB{public: voidsetC(int,int,int); voidshowC()const;privateconst: intc;};单继承与多继承2021/7/26104多继承举例classA{private:单继承与多继承2voidA::setA(intx){ a=x;}voidB::setB(intx){ b=x;}voidC::setC(intx,inty,intz){ //派生类成员直接访问基类的 //公有成员 setA(x); setB(y); c=z;}//其他函数实现略intmain(){ Cobj; obj.setA(5); obj.showA(); obj.setC(6,7,9); obj.showC();//obj.setB(6);错误//obj.showB();错误 return0;}1052021/7/26105voidA::setA(intx){intmain继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己的构造函数。定义构造函数时,只需要对本类中新增成员进行初始化,对继承来的基类成员的初始化,自动调用基类构造函数完成。派生类的构造函数需要给基类的构造函数传递参数派生类的构造、析构函数2021/7/26106继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己单一继承时的构造函数派生类名::派生类名(基类所需的形参,本类成员所需的形参):基类名(参数表){
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/26107单一继承时的构造函数派生类名::派生类名(基类所需的形参,本单一继承时的构造函数举例#include<iostream>usingnamecpacestd;classB{public: B(); B(inti); ~B(); voidprint()const;private: intb;};派生类的构造、析构函数2021/7/26108单一继承时的构造函数举例#include<iostream>B::B(){ b=0; cout<<"B'sdefaultconstructorcalled."<<endl;}B::B(inti){ b=i; cout<<"B'sconstructorcalled."<<endl;}B::~B(){ cout<<"B'sdestructorcalled."<<endl;}voidB::print()const{ cout<<b<<endl;}1092021/7/26109B::B(){342021/7/2634classC:publicB{public: C(); C(inti,intj); ~C(); voidprint()const;private: intc;};1102021/7/26110classC:publicB{352021/7/26C::C(){ c=0; cout<<"C'sdefaultconstructorcalled."<<endl;}C::C(inti,intj):B(i){ c=j; cout<<"C'sconstructorcalled."<<endl;}C::~C(){ cout<<"C'sdestructorcalled."<<endl;}voidC::print()const{ B::print(); cout<<c<<endl;}intmain(){ Cobj(5,6); obj.print(); return0;}1112021/7/26111C::C(){362021/7/2636多继承时的构造函数派生类名::派生类名(参数表):基类名1(基类1初始化参数表),基类名2(基类2初始化参数表),...基类名n(基类n初始化参数表){
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/26112多继承时的构造函数派生类名::派生类名(参数表):基类名1(派生类与基类的构造函数当基类中声明有缺省构造函数或未声明构造函数时,派生类构造函数可以不向基类构造函数传递参数,也可以不声明,构造派生类的对象时,基类的缺省构造函数将被调用。当需要执行基类中带形参的构造函数来初始化基类数据时,派生类构造函数应在初始化列表中为基类构造函数提供参数。派生类的构造、析构函数2021/7/26113派生类与基类的构造函数当基类中声明有缺省构造函数或未声明构造多继承且有内嵌对象时
的构造函数派生类名::派生类名(形参表):基类名1(参数),基类名2(参数),...基类名n(参数),新增成员对象的初始化{
本类成员初始化赋值语句;};派生类的构造、析构函数2021/7/26114多继承且有内嵌对象时
的构造函数派生类名::派生类名(形参表构造函数的执行顺序1.调用基类构造函数,调用顺序按照它们被继承时声明的顺序(从左向右)。2.对成员对象进行初始化,初始化顺序按照它们在类中声明的顺序。3.执行派生类的构造函数体中的内容。派生类的构造、析构函数2021/7/26115构造函数的执行顺序1.调用基类构造函数,调用顺序按
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2025年中职船舶与海洋工程装备(船舶导航)试题及答案
- 2025年大学化学(分析化学)试题及答案
- 2025年高职(美容美体艺术)美体塑形阶段测试题及答案
- 禁毒宣传知识课件
- 当前医患关系矛盾焦点
- 神秘杀手信息技术
- 祛斑方法培训课件
- 天津市第九中学2025-2026学年高三上学期12月月考语文试题(含答案)
- 2022-2023学年广东深圳罗湖区九年级上学期11月考英语试题含答案
- 2026广东清远市连南瑶族自治县聚汇劳务派遣有限公司第二次招聘门卫岗位1人备考题库及答案详解(夺冠系列)
- 临床成人吞咽障碍患者口服给药护理
- 儿童呼吸道合胞病毒感染诊断治疗和预防专家共识 4
- 雨课堂在线学堂《大数据技术与应用》作业单元考核答案
- 全国计算机等级考试一级WPS Office真题题库及答案
- 养牛场消防知识培训
- 义警法律知识培训总结课件
- 实施指南(2025)《DZT 0462.5-2023 矿产资源“三率”指标要求 第 5 部分:金、银、铌、钽、锂、锆、锶、稀土、锗》解读
- 棉尘安全培训课件
- 梯子作业安全培训效果课件
- 管控人力成本课件
- 吸附解析塔拆除施工方案
评论
0/150
提交评论