




版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
第七章继承与派生C++语言程序设计1第七章继承与派生C++语言程序设计1本章主要内容类的继承类成员的访问控制单继承与多继承派生类的构造、析构函数类成员的标识与访问2本章主要内容类的继承2类的继承与派生保持已有类的特性而构造新类的过程称为继承。在已有类的基础上新增自己的特性而产生新类的过程称为派生。被继承的已有类称为基类(或父类)。派生出的新类称为派生类。类的继承与派生3类的继承与派生保持已有类的特性而构造新类的过程称为继承。类的继承与派生问题举例类的继承与派生4继承与派生问题举例类的继承与派生4继承与派生的目的继承的目的:实现代码重用。派生的目的:当新的问题出现,原有程序无法解决(或不能完全解决)时,需要对原有程序进行改造。类的继承与派生5继承与派生的目的继承的目的:实现代码重用。类的继承与派生5派生类的声明class派生类名:继承方式基类名1{成员声明;}类的继承与派生6派生类的声明class派生类名:继承方式基类名1类的继承方式不同继承方式的影响主要体现在:派生类成员对基类成员的访问权限通过派生类对象对基类成员的访问权限三种继承方式公有继承私有继承保护继承类成员的访问控制7继承方式不同继承方式的影响主要体现在:类成员的访问控制7公有继承(public)基类的public和protected成员的访问属性在派生类中保持不变。(即:派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象只能访问基类的public成员。类成员的访问控制8公有继承(public)基类的public和protecte例7-1公有继承举例classPoint
{public: voidInitP(floatxx=0,floatyy=0){X=xx;Y=yy;} voidMove(floatxOff,floatyOff){X+=xOff;Y+=yOff;} floatGetX(){returnX;} floatGetY(){returnY;}private: floatX,Y;};类成员的访问控制9例7-1公有继承举例classPoint 类成员的访问控classRectangle:publicPoint
{public:voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};类成员的访问控制10classRectangle:publicPoint#include<iostream>usingnamespacestd;voidmain(){Rectanglerect;rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetW()<<','<<rect.GetH();}类成员的访问控制继承的方法11#include<iostream>类成员的访问控制继承的方私有继承(private)基类的public和protected成员都以private身份出现在派生类中。(即派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制12私有继承(private)基类的public和protect例7-2私有继承举例类成员的访问控制classRectangle:privatePoint
{public:voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};13例7-2私有继承举例类成员的访问控制classRecta#include<iostream>usingnamespacestd;voidmain(){Rectanglerect; rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetH()<<','<<rect.GetW();}类成员的访问控制非法访问14#include<iostream>类成员的访问控制非法访问
classRectangle:privatePoint {public: voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;}
voidMove(floatxOff,floatyOff){
Point::Move(xOff,yOff);} floatGetX(){returnPoint::GetX();}
floatGetY(){returnPoint::GetY();} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};类成员的访问控制15classRectangle:privatePoi#include<iostream>usingnamespacestd;voidmain(){Rectanglerect; rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetH()<<','<<rect.GetW();}类成员的访问控制访问派生类自身的方法16#include<iostream>类成员的访问控制访问派生保护继承(protected)基类的public和protected成员都以protected身份出现在派生类中。(即派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制17保护继承(protected)基类的public和prote#include<iostream>usingnamespacestd;classA{protected:intx;};classB:protectedA{public:voidFunction();};classC:privateB{public:voidFunction();};voidB::Function(){x=5;cout<<x<<endl;}voidC::Function(){x=15;cout<<x<<endl;}voidmain(){Bb;b.Function();Cc;c.Function();}18#include<iostream>voidB::Func#include<iostream>usingnamespacestd;classA{protected:intx;};classB:protectedA{public:voidFunction();};classC:privateB{public:voidFunction();};voidB::Function(){x=5;cout<<x<<endl;}voidC::Function(){x=15;cout<<x<<endl;}voidmain(){Bb;b.Function();Cc;c.Function();}protectedprivatex=15;cout<<x<<endl;19#include<iostream>voidB::Funcpublicprotectedprivatepublicpublicprotected隔离protectedprotectedprotected隔离privateprivateprivate隔离基类访问属性继承方式基类成员在派生类中的访问控制属性20publicprotectedprivatepublicpu类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象,反之则禁止。具体表现在:派生类的对象可以赋值给基类对象。派生类的对象可以初始化基类的引用。派生类对象的地址可以赋值给指向基类的指针。替代之后,派生类对象就可以作为基类的对象使用,但只能使用从基类继承的成员。派生类对象发挥基类对象的作用。类型兼容21类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象类型兼容#include<iostream>usingnamespacestd;classB0{public:voiddisplay(){cout<<"B0::display()"<<endl;} };classB1:publicB0
{public:voiddisplay(){cout<<"B1::display()"<<endl;}voidoutput(){cout<<"classB1B1::display()"<<endl;}};22类型兼容#include<iostream>22类型兼容voidmain() {B0b0; B1b1; b0.display();b0=b1;b0.display();b1.display();b0.output();b1.output();}b0.display();B0::display()b0.display();b1.display();b0.output();b1.output();B0::display()B1::display()编译错误classB1B1::display()P219例7-423类型兼容voidmain() b0.display();B基类与派生类的对应关系单继承派生类只从一个基类派生。多继承派生类从多个基类派生。多重派生由一个基类派生出多个不同的派生类。多层派生派生类又作为基类,继续派生新的类。单继承与多继承24基类与派生类的对应关系单继承单继承与多继承24多继承时派生类的声明class派生类名:继承方式1基类名1,继承方式2基类名2,...{成员声明;}注意:每一个“继承方式”,只用于限制其后的基类。单继承与多继承25多继承时派生类的声明class派生类名:继承方式1基类classA{public:voidsetA(int);voidshowA();private:inta;};classB{public:voidsetB(int);voidshowB();private:intb;};classC:publicA,privateB{public:voidsetC(int,int,int);voidshowC();private:intc;};单继承与多继承26classAclassC:publicA,privoidA::setA(intx){a=x;}voidB::setB(intx){b=x;}voidC::setC(intx,inty,intz){setA(x);
setB(y);c=z;}//其他函数实现略voidmain(){Cobj;obj.setA(5);obj.showA();obj.setC(6,7,9);obj.showC();
obj.setB(6);obj.showB();
}obj.setB(6);obj.showB();27voidA::setA(intx)voidmain(继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己的构造函数。声明构造函数时,只需要对本类中新增成员进行初始化,对继承来的基类成员的初始化,自动调用基类构造函数完成。派生类的构造、析构函数28继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己继承时的构造函数派生类的构造、析构函数当基类中有默认形式的构造函数或带默认形参值的构造函数时,派生类构造函数可以不向基类构造函数传递参数。若基类中未声明构造函数,派生类中也可以不声明,全采用默认形式构造函数。当基类声明有带形参的构造函数时,派生类也应声明带形参的构造函数,并将参数传递给基类构造函数。29继承时的构造函数派生类的构造、析构函数当基类中有默认形式的构单一继承时的构造函数派生类的构造、析构函数classB{B();B(inti);…}classC:publicB{C();C(inti,intj)}…派生类C的构造函数有以下几种实现方式:C::C(){}或C::C(inti,intj){…}表示先调用B类默认的构造函数或带默认形参值的构造函数初始化B类数据成员。C::C(inti,intj):B(i){…}表示先调用B类带参数的构造函数初始化B类数据成员。30单一继承时的构造函数派生类的构造、析构函数classB{#include<iostream>usingnamespacestd;classB{public:B();B(inti); private:intb;};classC:publicB{public:C();C(inti,intj);private:intc;};B::B(){b=0;cout<<"B'sdefaultconstructorcalled."<<endl;}B::B(inti){b=i;cout<<"B'sconstructorcalled."<<b<<endl;}C::C(){c=0;cout<<"C'sdefaultconstructorcalled."<<endl;}C::C(inti,intj):B(i){c=j;cout<<"C'sconstructorcalled."<<c<<endl;}voidmain(){Cobj(3,5);}31#include<iostream>B::B()void多继承时的构造函数派生类名::派生类名(参数总表):基类名1(参数),基类名2(参数),...,基类名n(参数),对象数据成员的初始化{本类成员初始化赋值语句;};派生类的构造、析构函数32多继承时的构造函数派生类名::派生类名(参数总表):派生类构造函数的调用顺序1.调用基类构造函数,调用顺序按照它们被继承时声明的顺序(从左向右)。2.调用成员对象的构造函数,调用顺序按照它们在类中声明的顺序。3.派生类的构造函数体中的内容。P222例7-5派生类的构造、析构函数33构造函数的调用顺序1.调用基类构造函数,调用顺序按照它们被拷贝构造函数若建立派生类对象时调用默认拷贝构造函数,则编译器将自动调用基类的默认拷贝构造函数。若编写派生类的拷贝构造函数,则需要为基类相应的拷贝构造函数传递参数。例如:C::C(C&c1):B(c1){…}派生类的构造、析构函数34拷贝构造函数若建立派生类对象时调用默认拷贝构造函数,则编译器派生类的构造、析构函数#include<iostream>usingnamespacestd;classB{public:B(inti);B(B&);~B(); private:intb;};classC:publicB{public:C(inti,intj);C(C&);~C(); private:intc;};35派生类的构造、析构函数#include<iostream>336派生类的构造、析构函数B::B(B&b1){b=0;cout<<"B'scopyconstructorcalled."<<endl;}B::B(inti){b=i;cout<<"B'sconstructorcalled."<<b<<endl;}B::~B(){cout<<"B'sdestructorcalled."<<endl;}3636派生类的构造、析构函数B::B(B&b1)3637派生类的构造、析构函数C::C(C&c1):B(c1){c=0;cout<<"C'scopyconstructorcalled."<<endl;}C::C(inti,intj):B(i){c=j;cout<<"C'sconstructorcalled."<<c<<endl;}C::~C(){cout<<"C'sdestructorcalled."<<endl;}voidmain(){Cobj(3,5);Cobj2(obj);}3737派生类的构造、析构函数C::C(C&c1):B(c1)38派生类的构造、析构函数3838派生类的构造、析构函数38继承时的析构函数析构函数也不被继承,派生类自行声明声明方法与一般(无继承关系时)类的析构函数相同。不需要显式地调用基类的析构函数,系统会自动隐式调用。析构函数的调用次序与构造函数相反。P224例7-6派生类的构造、析构函数39继承时的析构函数析构函数也不被继承,派生类自行声明派生类的构例7-6派生类析构函数举例派生类的构造、析构函数#include<iostream>usingnamespacestd;classB1 //基类B1声明{public: B1(inti){cout<<"constructingB1"<<i<<endl;}
~B1(){cout<<"destructingB1"<<endl;}};classB2 //基类B2声明{public: B2(intj){cout<<"constructingB2"<<j<<endl;}
~B2(){cout<<"destructingB2"<<endl;}
};classB3 //基类B3声明{public: B3(){cout<<"constructingB3*"<<endl;}
~B3(){cout<<"destructingB3"<<endl;}
};40例7-6派生类析构函数举例派生类的构造、析构函数#incclassC:publicB2,publicB1,publicB3 {public: C(inta,intb,intc,intd):B1(a),memberB2(d),memberB1(c),B2(b){}private: B1memberB1; B2memberB2; B3memberB3;};voidmain(){ Cobj(1,2,3,4);}41classC:publicB2,publicB1,例7-6运行结果constructingB22constructingB11constructingB3*constructingB13constructingB24constructingB3*destructingB3destructingB2destructingB1destructingB3destructingB1destructingB242例7-6运行结果constructingB2242同名隐藏规则当派生类与基类中有相同成员时:若未强行指名,则通过派生类对象使用的是派生类中的同名成员。如果派生类中声明了与基类成员函数同名的新函数,即使函数的参数表不同,从基类继承的同名函数的所有重载形式也都会被隐藏。如果某派生类的多个基类拥有同名的成员,同时,派生类又新增这样的同名成员,派生类成员将隐藏所有基类的同名成员。如要通过派生类对象访问基类中被覆盖的同名成员,应使用基类名限定。派生类成员的标识与访问43同名隐藏规则当派生类与基类中有相同成员时:派生类成员的标识与例7-7多继承同名隐藏举例派生类成员的标识与访问#include<iostream>usingnamespacestd;classB1 {public: intnV; voidfun(){cout<<"MemberofB1"<<nV<<endl;}};classB2 {public: intnV; voidfun(){cout<<"MemberofB2"<<nV<<endl;}};44例7-7多继承同名隐藏举例派生类成员的标识与访问#incclassD1:publicB1,publicB2 {public: intnV; voidfun(){cout<<"MemberofD1"<<nV<<endl;}};voidmain(){ D1d1;
d1.nV=1;//对象名.成员名标识,访问D1类成员
d1.fun();
d1.B1::nV=2;//作用域分辨符标识,访问基类B1成员
d1.B1::fun();
d1.B2::nV=3;//作用域分辨符标识,访问基类B2成员
d1.B2::fun(); }派生类成员的标识与访问45classD1:publicB1,publicB2二义性问题举例(一)classA{public:voidf();};classB{public:voidf();voidg()};classC:publicA,publicB{public:voidg();voidh();};如果声明:Cc1;则c1.f();具有二义性而c1.g();无二义性
(同名覆盖)派生类成员的标识与访问46二义性问题举例(一)classAclassC:publ二义性的解决方法解决方法一:用类名来限定
c1.A::f()或c1.B::f()解决方法二:同名覆盖
在C中声明一个同名成员函数f(),f()再根据需要调用A::f()或B::f()派生类成员的标识与访问47二义性的解决方法解决方法一:用类名来限定
c1.A::f()二义性问题举例(二)classB{public:intb;};classB1:publicB{private:intb1;};classB2:publicB{private:intb2;};classC:publicB1,publicB2{public:intf();private:intd;};派生类成员的标识与访问48二义性问题举例(二)classBclassC:pub派生类C的对象的存储结构示意图:bb1bb2dB类成员B类成员B1类成员B2类成员C类对象有二义性:Cc;c.bc.B::b无二义性:c.B1::bc.B2::b派生类成员的标识与访问49派生类C的对象的存储结构示意图:bb1bb2dB类成员B类成二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成员时,将出现访问时的二义性(不确定性)。当派生类从多个基类派生,而这些基类又从同一个基类派生,则在访问此共同基类中的成员时,将产生二义性。派生类成员的标识与访问50二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成虚基类虚基类的引入用于有共同基类的场合声明以virtual修饰说明基类
例:classB1:virtualpublicB作用主要用来解决多继承时可能发生的对同一基类继承多次而产生的二义性问题。为最远的派生类提供惟一的基类成员,而不重复产生多次拷贝。注意在第一级继承时就要将共同基类设计为虚基类。虚基类51虚基类虚基类的引入虚基类51虚基类举例classB{public:intb;};classB1:virtualpublicB{private:intb1;};classB2:virtualpublicB{private:intb2;};classC:publicB1,publicB2{private:floatd;}下面的访问是正确的:Ccobj;cobj.b;虚基类52虚基类举例classB{public:intb;};虚基类的派生类对象存储结构示意图:BB1B2Cb1b2dB1类成员B2类成员C类对象bB类成员虚基类53虚基类的派生类对象存储结构示意图:BB1B2Cb1b2dB1例7-8虚基类举例虚基类D1nV:intnVd:intB1::nV1:intB2::nV2:intfund():voidfun():voidB1nV1:intB2nV2:intD1nVd:intfund():void<<virtual>>B0nV:intfun()54例7-8虚基类举例虚基类D1nV:B0B1新增成员B0B2新增成员D1新增成员B0B0B1B2D1nV,fun()虚基类55B0B1新增成员B0B2新增成员D1新增成员B0B0B1B2#include<iostream>usingnamespacestd;classB0 {public: intnV; voidfun(){cout<<"MemberofB0"<<endl;}};classB1:virtualpublicB0{public: intnV1;};classB2:virtualpublicB0{public: intnV2;};虚基类56#include<iostream>虚基classD1:publicB1,publicB2 {public: //新增外部接口 intnVd; voidfund(){cout<<"MemberofD1"<<endl;}};voidmain() {D1d1; //声明D1类对象d1 d1.nV=2; d1.fun();}虚基类57classD1:publicB1,publicB2虚基类及其派生类构造函数建立对象时所指定的类称为最(远)派生类。虚基类的成员是由最远派生类的构造函数通过调用虚基类的构造函数进行初始化的。在整个继承结构中,直接或间接继承虚基类的所有派生类,都必须在构造函数的成员初始化表中给出对虚基类的构造函数的调用。如果未列出,则表示调用该虚基类的默认构造函数。在建立对象时,只有最远派生类的构造函数调用虚基类的构造函数,该派生类的其他基类对虚基类构造函数的调用被忽略。虚基类58虚基类及其派生类构造函数建立对象时所指定的类称为最(远)派生虚基类#include<iostream>usingnamespacestd;classB0 {public: B0(intn){nV=n;} intnV; voidfun(){cout<<"MemberofB0"<<endl;}};classB1:virtualpublicB0 {public: B1(inta):B0(a){} intnV1;};classB2:virtualpublicB0 {public: B2(inta):B0(a){} intnV2;};P23459虚基类#include<iostrclassD1:publicB1,publicB2{public: D1(inta):B0(a),B1(a),B2(a){} intnVd; voidfund(){cout<<"MemberofD1"<<endl;}};voidmain() { D1d1(1); d1.nV=2; d1.fun();}60classD1:publicB1,publicB2综合实例例7-10(课后阅读)这个程序有两点不足:①基类的成员函数pay()的函数体为空,在实现部分仍要写出函数体,显得冗余。②在main()函数中,建立了四个不同类的对象,对它们进行了类似的操作,但是却重复写了四遍类似的语句,程序不够简洁。61综合实例例7-10(课后阅读)61练习1:设计一个类CDateInfo,要求其满足下述要求。(1)要求有一个无参的构造函数,其初始的年,月,日分别为:2000,1,1。(2)要求有一个带参数的构造函数,其参数分别对应年,月,日。(3)要求用一个成员函数实现日期的设置。(4)要求用一个成员函数实现日期的获取。练习2:定义基类BaseClass,有1个公有成员函数gcd(intx,inty),其功能是求两个数的最大公约数。由BaseClass私有派生出Derived类,在main函数中用Derived类的对象中调用基类函数gcd(intx,inty),求任意两个整数的最大公约数。62练习1:设计一个类CDateInfo,要求其满足下述要求。6小结与复习建议主要内容类的继承、类成员的访问控制、单继承与多继承、派生类的构造和析构函数、类成员的标识与访问达到的目标理解类的继承关系,学会使用继承关系实现代码的重用。实验任务实验七63小结与复习建议主要内容63第七章继承与派生C++语言程序设计64第七章继承与派生C++语言程序设计1本章主要内容类的继承类成员的访问控制单继承与多继承派生类的构造、析构函数类成员的标识与访问65本章主要内容类的继承2类的继承与派生保持已有类的特性而构造新类的过程称为继承。在已有类的基础上新增自己的特性而产生新类的过程称为派生。被继承的已有类称为基类(或父类)。派生出的新类称为派生类。类的继承与派生66类的继承与派生保持已有类的特性而构造新类的过程称为继承。类的继承与派生问题举例类的继承与派生67继承与派生问题举例类的继承与派生4继承与派生的目的继承的目的:实现代码重用。派生的目的:当新的问题出现,原有程序无法解决(或不能完全解决)时,需要对原有程序进行改造。类的继承与派生68继承与派生的目的继承的目的:实现代码重用。类的继承与派生5派生类的声明class派生类名:继承方式基类名1{成员声明;}类的继承与派生69派生类的声明class派生类名:继承方式基类名1类的继承方式不同继承方式的影响主要体现在:派生类成员对基类成员的访问权限通过派生类对象对基类成员的访问权限三种继承方式公有继承私有继承保护继承类成员的访问控制70继承方式不同继承方式的影响主要体现在:类成员的访问控制7公有继承(public)基类的public和protected成员的访问属性在派生类中保持不变。(即:派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象只能访问基类的public成员。类成员的访问控制71公有继承(public)基类的public和protecte例7-1公有继承举例classPoint
{public: voidInitP(floatxx=0,floatyy=0){X=xx;Y=yy;} voidMove(floatxOff,floatyOff){X+=xOff;Y+=yOff;} floatGetX(){returnX;} floatGetY(){returnY;}private: floatX,Y;};类成员的访问控制72例7-1公有继承举例classPoint 类成员的访问控classRectangle:publicPoint
{public:voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};类成员的访问控制73classRectangle:publicPoint#include<iostream>usingnamespacestd;voidmain(){Rectanglerect;rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetW()<<','<<rect.GetH();}类成员的访问控制继承的方法74#include<iostream>类成员的访问控制继承的方私有继承(private)基类的public和protected成员都以private身份出现在派生类中。(即派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制75私有继承(private)基类的public和protect例7-2私有继承举例类成员的访问控制classRectangle:privatePoint
{public:voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};76例7-2私有继承举例类成员的访问控制classRecta#include<iostream>usingnamespacestd;voidmain(){Rectanglerect; rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetH()<<','<<rect.GetW();}类成员的访问控制非法访问77#include<iostream>类成员的访问控制非法访问
classRectangle:privatePoint {public: voidInitR(floatx,floaty,floatw,floath){InitP(x,y);W=w;H=h;}
voidMove(floatxOff,floatyOff){
Point::Move(xOff,yOff);} floatGetX(){returnPoint::GetX();}
floatGetY(){returnPoint::GetY();} floatGetH(){returnH;} floatGetW(){returnW;}private: floatW,H;};类成员的访问控制78classRectangle:privatePoi#include<iostream>usingnamespacestd;voidmain(){Rectanglerect; rect.InitR(2,3,20,10); rect.Move(3,2); cout<<rect.GetX()<<','<<rect.GetY()<<',' <<rect.GetH()<<','<<rect.GetW();}类成员的访问控制访问派生类自身的方法79#include<iostream>类成员的访问控制访问派生保护继承(protected)基类的public和protected成员都以protected身份出现在派生类中。(即派生类中的成员函数可以直接访问基类中的public和protected成员)派生类的成员函数不能直接访问基类的private成员。通过派生类的对象不能直接访问基类中的任何成员。类成员的访问控制80保护继承(protected)基类的public和prote#include<iostream>usingnamespacestd;classA{protected:intx;};classB:protectedA{public:voidFunction();};classC:privateB{public:voidFunction();};voidB::Function(){x=5;cout<<x<<endl;}voidC::Function(){x=15;cout<<x<<endl;}voidmain(){Bb;b.Function();Cc;c.Function();}81#include<iostream>voidB::Func#include<iostream>usingnamespacestd;classA{protected:intx;};classB:protectedA{public:voidFunction();};classC:privateB{public:voidFunction();};voidB::Function(){x=5;cout<<x<<endl;}voidC::Function(){x=15;cout<<x<<endl;}voidmain(){Bb;b.Function();Cc;c.Function();}protectedprivatex=15;cout<<x<<endl;82#include<iostream>voidB::Funcpublicprotectedprivatepublicpublicprotected隔离protectedprotectedprotected隔离privateprivateprivate隔离基类访问属性继承方式基类成员在派生类中的访问控制属性83publicprotectedprivatepublicpu类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象,反之则禁止。具体表现在:派生类的对象可以赋值给基类对象。派生类的对象可以初始化基类的引用。派生类对象的地址可以赋值给指向基类的指针。替代之后,派生类对象就可以作为基类的对象使用,但只能使用从基类继承的成员。派生类对象发挥基类对象的作用。类型兼容84类型兼容规则一个公有派生类的对象在使用上可以被当作基类的对象类型兼容#include<iostream>usingnamespacestd;classB0{public:voiddisplay(){cout<<"B0::display()"<<endl;} };classB1:publicB0
{public:voiddisplay(){cout<<"B1::display()"<<endl;}voidoutput(){cout<<"classB1B1::display()"<<endl;}};85类型兼容#include<iostream>22类型兼容voidmain() {B0b0; B1b1; b0.display();b0=b1;b0.display();b1.display();b0.output();b1.output();}b0.display();B0::display()b0.display();b1.display();b0.output();b1.output();B0::display()B1::display()编译错误classB1B1::display()P219例7-486类型兼容voidmain() b0.display();B基类与派生类的对应关系单继承派生类只从一个基类派生。多继承派生类从多个基类派生。多重派生由一个基类派生出多个不同的派生类。多层派生派生类又作为基类,继续派生新的类。单继承与多继承87基类与派生类的对应关系单继承单继承与多继承24多继承时派生类的声明class派生类名:继承方式1基类名1,继承方式2基类名2,...{成员声明;}注意:每一个“继承方式”,只用于限制其后的基类。单继承与多继承88多继承时派生类的声明class派生类名:继承方式1基类classA{public:voidsetA(int);voidshowA();private:inta;};classB{public:voidsetB(int);voidshowB();private:intb;};classC:publicA,privateB{public:voidsetC(int,int,int);voidshowC();private:intc;};单继承与多继承89classAclassC:publicA,privoidA::setA(intx){a=x;}voidB::setB(intx){b=x;}voidC::setC(intx,inty,intz){setA(x);
setB(y);c=z;}//其他函数实现略voidmain(){Cobj;obj.setA(5);obj.showA();obj.setC(6,7,9);obj.showC();
obj.setB(6);obj.showB();
}obj.setB(6);obj.showB();90voidA::setA(intx)voidmain(继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己的构造函数。声明构造函数时,只需要对本类中新增成员进行初始化,对继承来的基类成员的初始化,自动调用基类构造函数完成。派生类的构造、析构函数91继承时的构造函数基类的构造函数不被继承,派生类中需要声明自己继承时的构造函数派生类的构造、析构函数当基类中有默认形式的构造函数或带默认形参值的构造函数时,派生类构造函数可以不向基类构造函数传递参数。若基类中未声明构造函数,派生类中也可以不声明,全采用默认形式构造函数。当基类声明有带形参的构造函数时,派生类也应声明带形参的构造函数,并将参数传递给基类构造函数。92继承时的构造函数派生类的构造、析构函数当基类中有默认形式的构单一继承时的构造函数派生类的构造、析构函数classB{B();B(inti);…}classC:publicB{C();C(inti,intj)}…派生类C的构造函数有以下几种实现方式:C::C(){}或C::C(inti,intj){…}表示先调用B类默认的构造函数或带默认形参值的构造函数初始化B类数据成员。C::C(inti,intj):B(i){…}表示先调用B类带参数的构造函数初始化B类数据成员。93单一继承时的构造函数派生类的构造、析构函数classB{#include<iostream>usingnamespacestd;classB{public:B();B(inti); private:intb;};classC:publicB{public:C();C(inti,intj);private:intc;};B::B(){b=0;cout<<"B'sdefaultconstructorcalled."<<endl;}B::B(inti){b=i;cout<<"B'sconstructorcalled."<<b<<endl;}C::C(){c=0;cout<<"C'sdefaultconstructorcalled."<<endl;}C::C(inti,intj):B(i){c=j;cout<<"C'sconstructorcalled."<<c<<endl;}voidmain(){Cobj(3,5);}94#include<iostream>B::B()void多继承时的构造函数派生类名::派生类名(参数总表):基类名1(参数),基类名2(参数),...,基类名n(参数),对象数据成员的初始化{本类成员初始化赋值语句;};派生类的构造、析构函数95多继承时的构造函数派生类名::派生类名(参数总表):派生类构造函数的调用顺序1.调用基类构造函数,调用顺序按照它们被继承时声明的顺序(从左向右)。2.调用成员对象的构造函数,调用顺序按照它们在类中声明的顺序。3.派生类的构造函数体中的内容。P222例7-5派生类的构造、析构函数96构造函数的调用顺序1.调用基类构造函数,调用顺序按照它们被拷贝构造函数若建立派生类对象时调用默认拷贝构造函数,则编译器将自动调用基类的默认拷贝构造函数。若编写派生类的拷贝构造函数,则需要为基类相应的拷贝构造函数传递参数。例如:C::C(C&c1):B(c1){…}派生类的构造、析构函数97拷贝构造函数若建立派生类对象时调用默认拷贝构造函数,则编译器派生类的构造、析构函数#include<iostream>usingnamespacestd;classB{public:B(inti);B(B&);~B(); private:intb;};classC:publicB{public:C(inti,intj);C(C&);~C(); private:intc;};98派生类的构造、析构函数#include<iostream>399派生类的构造、析构函数B::B(B&b1){b=0;cout<<"B'scopyconstructorcalled."<<endl;}B::B(inti){b=i;cout<<"B'sconstructorcalled."<<b<<endl;}B::~B(){cout<<"B'sdestructorcalled."<<endl;}9936派生类的构造、析构函数B::B(B&b1)36100派生类的构造、析构函数C::C(C&c1):B(c1){c=0;cout<<"C'scopyconstructorcalled."<<endl;}C::C(inti,intj):B(i){c=j;cout<<"C'sconstructorcalled."<<c<<endl;}C::~C(){cout<<"C'sdestructorcalled."<<endl;}voidmain(){Cobj(3,5);Cobj2(obj);}10037派生类的构造、析构函数C::C(C&c1):B(c1)101派生类的构造、析构函数10138派生类的构造、析构函数38继承时的析构函数析构函数也不被继承,派生类自行声明声明方法与一般(无继承关系时)类的析构函数相同。不需要显式地调用基类的析构函数,系统会自动隐式调用。析构函数的调用次序与构造函数相反。P224例7-6派生类的构造、析构函数102继承时的析构函数析构函数也不被继承,派生类自行声明派生类的构例7-6派生类析构函数举例派生类的构造、析构函数#include<iostream>usingnamespacestd;classB1 //基类B1声明{public: B1(inti){cout<<"constructingB1"<<i<<endl;}
~B1(){cout<<"destructingB1"<<endl;}};classB2 //基类B2声明{public: B2(intj){cout<<"constructingB2"<<j<<endl;}
~B2(){cout<<"destructingB2"<<endl;}
};classB3 //基类B3声明{public: B3(){cout<<"constructingB3*"<<endl;}
~B3(){cout<<"destructingB3"<<endl;}
};103例7-6派生类析构函数举例派生类的构造、析构函数#incclassC:publicB2,publicB1,publicB3 {public: C(inta,intb,intc,intd):B1(a),memberB2(d),memberB1(c),B2(b){}private: B1memberB1; B2memberB2; B3memberB3;};voidmain(){ Cobj(1,2,3,4);}104classC:publicB2,publicB1,例7-6运行结果constructingB22constructingB11constructingB3*constructingB13constructingB24constructingB3*destructingB3destructingB2destructingB1destructingB3destructingB1destructingB2105例7-6运行结果constructingB2242同名隐藏规则当派生类与基类中有相同成员时:若未强行指名,则通过派生类对象使用的是派生类中的同名成员。如果派生类中声明了与基类成员函数同名的新函数,即使函数的参数表不同,从基类继承的同名函数的所有重载形式也都会被隐藏。如果某派生类的多个基类拥有同名的成员,同时,派生类又新增这样的同名成员,派生类成员将隐藏所有基类的同名成员。如要通过派生类对象访问基类中被覆盖的同名成员,应使用基类名限定。派生类成员的标识与访问106同名隐藏规则当派生类与基类中有相同成员时:派生类成员的标识与例7-7多继承同名隐藏举例派生类成员的标识与访问#include<iostream>usingnamespacestd;classB1 {public: intnV; voidfun(){cout<<"MemberofB1"<<nV<<endl;}};classB2 {public: intnV; voidfun(){cout<<"MemberofB2"<<nV<<endl;}};107例7-7多继承同名隐藏举例派生类成员的标识与访问#incclassD1:publicB1,publicB2 {public: intnV; voidfun(){cout<<"MemberofD1"<<nV<<endl;}};voidmain(){ D1d1;
d1.nV=1;//对象名.成员名标识,访问D1类成员
d1.fun();
d1.B1::nV=2;//作用域分辨符标识,访问基类B1成员
d1.B1::fun();
d1.B2::nV=3;//作用域分辨符标识,访问基类B2成员
d1.B2::fun(); }派生类成员的标识与访问108classD1:publicB1,publicB2二义性问题举例(一)classA{public:voidf();};classB{public:voidf();voidg()};classC:publicA,publicB{public:voidg();voidh();};如果声明:Cc1;则c1.f();具有二义性而c1.g();无二义性
(同名覆盖)派生类成员的标识与访问109二义性问题举例(一)classAclassC:publ二义性的解决方法解决方法一:用类名来限定
c1.A::f()或c1.B::f()解决方法二:同名覆盖
在C中声明一个同名成员函数f(),f()再根据需要调用A::f()或B::f()派生类成员的标识与访问110二义性的解决方法解决方法一:用类名来限定
c1.A::f()二义性问题举例(二)classB{public:intb;};classB1:publicB{private:intb1;};classB2:publicB{private:intb2;};classC:publicB1,publicB2{public:intf();private:intd;};派生类成员的标识与访问111二义性问题举例(二)classBclassC:pub派生类C的对象的存储结构示意图:bb1bb2dB类成员B类成员B1类成员B2类成员C类对象有二义性:Cc;c.bc.B::b无二义性:c.B1::bc.B2::b派生类成员的标识与访问112派生类C的对象的存储结构示意图:bb1bb2dB类成员B类成二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成员时,将出现访问时的二义性(不确定性)。当派生类从多个基类派生,而这些基类又从同一个基类派生,则在访问此共同基类中的成员时,将产生二义性。派生类成员的标识与访问113二义性问题在多继承时,基类与派生类之间,或基类之间出现同名成虚基类虚
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2025年交通设备制造业数字化转型升级政策环境分析报告
- 2025年工业互联网平台传感器网络自组网技术在航空航天领域的应用分析
- 2025年分布式能源系统生物质能源应用中的能源互联网发展优化报告
- 2025年乡村振兴背景下职业技能培训的可持续发展策略报告
- 2025年CCS项目在能源领域应用的经济效益与投资决策支持研究报告
- 2025年医疗美容消费者心理特点与服务质量优化路径报告
- 轻工行业25W22:关税博弈继续浆价震荡分化
- 施工净化车间管理制度
- 固体废物收集点管理制度
- 所属分公司财务管理制度
- 2025年江西省中考数学试卷真题(含标准答案)
- 2025年河北省中考麒麟卷生物(三)及答案
- 2025年河北省万唯中考定心卷地理(二)
- 2025年高考全国二卷英语高考真题含解析
- 有机化学所有的命名--超全.
- 引水罐的设计计算
- 三年级译林版英语下学期按要求写句子专项强化练习题
- 电缆接线工艺设计规范流程
- 中医经络减肥课件
- 5WHY分析法培训
- 巧克力糖自动包装机 课程设计
评论
0/150
提交评论