




版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、会计学1C备课讲义教程实用备课讲义教程实用2虚函数多态性:调用同一个函数名,可以根据需要但实现不同的功能。多态性是面向对象的程序设计的关键技术。编译时的多态性(函数重载)运行时的多态性(虚函数)多态性运行时的多态性是指在程序执行之前,根据函数名和参数无法确定应该调用哪一个函数,必须在程序的执行过程中,根据具体的执行情况来动态地确定第1页/共36页3可以将一个派生类对象的地址赋给基类的指针变量。基类对象派生类对象Base b;Derive d;Base *basep;basepbasep=&b;basepbasep = &d;basep只能引用从基类继承来的成员。xShow()xShow()yS
2、how()basep -Show();basep-Show()基类指针派生类对象基类对象第2页/共36页4class Pointfloat x,y;public:Point()Point(float i,float j)x=i;y=j;float area(void)return 0.0; ;const float Pi=3.14159;class Circle:public Point/类Point的派生类float radius;public:Circle(float r)radius=r;float area(void) return Pi*radius*radius;void main
3、(void) Point *pp; /基类指针,可以将派生类对象的地址赋给基类指针 Circle c(5.4321); pp=&c; coutarea ()Show()Base *basep;basep=&b;basep = &d;basep -Show();即指向派生类新增的成员函数需要将基类中的Show()说明为虚函数第4页/共36页6若要访问派生类中相同名字的函数,必须将基类中的同名函数定义为虚函数,这样,将不同的派生类对象的地址赋给基类的指针变量后,就可以动态地根据这种赋值语句调用不同类中的函数。第5页/共36页7class Point float x,y;public:Point()
4、Point(float i,float j)x=i;y=j;virtual float area(void) return 0.0; ;const float Pi=3.14159;class Circle:public Point/类Point的派生类float radius;public:Circle(float r)radius=r;float area(void) return Pi*radius*radius;void main(void) Point *pp; /基类指针,可以将派生类对象的地址赋给基类指针 Circle c(5.4321); pp=&c; coutarea ()e
5、ndl; /调用虚函数将area()声明为虚函数,编译器对其进行动态聚束,按照实际对象c调用了Circle中的函数area()。使Point类中的area()与Circle类中的area()有一个统一的接口。输出:92.7011声明为虚函数调用虚函数虚函数再定义第6页/共36页8虚函数的定义和使用 可以在程序运行时通过调用相同的函数名而实现不同功能的函数称为虚函数。定义格式为:virtual FuncName();一旦把基类的成员函数定义为虚函数,由基类所派生出来的所有派生类中,该函数均保持虚函数的特性。 在派生类中重新定义基类中的虚函数时,可以不用关键字virtual来修饰这个成员函数 。第
6、7页/共36页9虚函数是用关键字virtual修饰的某基类中的protected或public成员函数。它可以在派生类中重新定义,以形成不同版本。只有在程序的执行过程中,依据指针具体指向哪个类对象,或依据引用哪个类对象,才能确定激活哪一个版本,实现动态聚束。第8页/共36页10class Aprotected: int x;public:A()x =1000; virtual void print()cout “x=”xt; /虚函数;class B:public Aint y;public:B() y=2000;void print()cout “y=”yt; /派生虚函数;class C:
7、public Aint z;public:C()z=3000;void print()cout “z=”zprint();/调用类A的虚函数 pa=&b; pa-print();/调用类B的虚函数 pa=&c; pa-print();/调用类C的虚函数x=1000y=2000z=3000 x=1000y=2000z=3000第9页/共36页11class Base public : virtual int Set(int a, int b) . .;class Derive:public Basepublic : int Set(int x, int y) . .;class Base pub
8、lic : virtual int Set(int a, int b) . .;class Derive:public Basepublic : int Set(int x, int y=0) . .;int Set(int ,int )是虚函数两个Set()函数参数不一致,是重载,不是虚函数第10页/共36页12关于虚函数,说明以下几点:1、当在基类中把成员函数定义为虚函数后,在其派生类中定义的虚函数必须与基类中的虚函数同名,参数的类型、顺序、参数的个数必须一一对应,函数的返回的类型也相同。若函数名相同,但参数的个数不同或者参数的类型不同时,则属于函数的重载,而不是虚函数。若函数名不同,显然
9、这是不同的成员函数。第11页/共36页132、实现这种动态的多态性时,必须使用基类类型的指针变量或基类引用,并使该指针或引用指向不同的派生类对象,并通过调用指针或引用所指向的虚函数才能实现动态的多态性。xShow()xShow()yShow()xShow()zShow()类A类B类CShow()定义为虚函数类B与类C均为类A的公有派生。A *p; B b;C c; p=&b ; p-Show();p=&c ; p-Show();即在程序运行时,通过赋值语句实现多态性第12页/共36页143、虚函数必须是类的一个成员函数,不能是友元函数,也不能是静态的成员函数。4、在派生类中没有重新定义虚函数时
10、,与一般的成员函数一样,当调用这种派生类对象的虚函数时,则调用其基类中的虚函数。5、可把析构函数定义为虚函数,但是,不能将构造函数定义为虚函数。第13页/共36页156、虚函数与一般的成员函数相比较,调用时的执行速度要慢一些。为了实现多态性,在每一个派生类中均要保存相应虚函数的入口地址表,函数的调用机制也是间接实现的。因此,除了要编写一些通用的程序,并一定要使用虚函数才能完成其功能要求外,通常不必使用虚函数。7、一个函数如果被定义成虚函数,则不管经历多少次派生,仍将保持其虚特性,以实现“一个接口,多个形态”。第14页/共36页16虚函数的访问用基类指针或基类引用访问与用派生类对象名访问用基类指
11、针或基类引用访问虚函数时,指向其实际派生类对象重新定义的函数。实现动态聚束。通过一个派生类对象名访问时,只能静态聚束。即由编译器在编译的时候决定调用哪个函数。第15页/共36页17class Point float x,y;public:Point()Point(float i,float j)x=i;y=j;virtual float area(void) return 0.0; /声明为虚函数;const float Pi=3.14159;class Circle:public Point/类Point的派生类float radius;public:Circle(float r)radiu
12、s=r;float area(void) return Pi*radius*radius;/虚函数再定义;void main(void) Point *pp; /基类指针,可以将派生类对象的地址赋给基类指针 Circle c(5.4321); coutc.area()endl; coutc.Point:area()endl; coutc.Circle:area ()endl;输出:92.7011092.7011可见,利用对象名进行调用与一般非虚函数没有区别。用对象名调用area( )第16页/共36页18class base0public:void v(void)coutbase0n;clas
13、s base1:public base0public:virtual void v(void) coutbase1n; ;class A1:public base1public:void v()coutA1n;class A2:public A1public:void v(void)coutA2n;class B1:private base1public:void v(void)coutB1n;class B2:public B1public:void v(void)coutv(); A2 a2; (pb=&a2)-v(); B1 b1; (pb=&b1)-v(); B2 b2; (pb=&b
14、2)-v();base0base0私有派生,在类外不能调用基类函数第17页/共36页19class base0public:void v(void)coutbase0n;class base1:public base0public:virtual void v(void) coutbase1n; ;class A1:public base1public:void v()coutA1n;class A2:public A1public:void v(void)coutA2n;class B1:private base1public:void v(void)coutB1n;class B2:pub
15、lic B1public:void v(void)coutv(); A2 a2; (pb=&a2)-v();A1A2第18页/共36页20下面程序的输出是 。class Aprotected:int x;public:A()x =1000;virtual void p()cout x=xn; p2(); virtual void p2()coutA:p2()endl; ;class C:public Aint z;public:C()z=3000; void p()cout z=zn; p2();virtual void p2()coutC:p2()p();pa=&c;pa-p();x=100
16、0z=3000C:p2()A:p2()第19页/共36页21class B0public: virtual void display()cout B0:display()n;class B1:public B0public: void display()cout B1:display()n;class D:public B1public: void display()cout display();void main() B0 b0, *p; p = &b0; fun(p); B1 b1; p = &b1; fun(p); D d; p = &d; fun(p);B0:display()B1:d
17、isplay()D:display()第20页/共36页22class B0public: virtual void display()cout B0:display()n;class B1:public B0public: void display()cout B1:display()n;class D:public B1public: void display()cout D:display()n;void fun(B0 &bb) bb.display();void main() B0 b0; fun(b0); B1 b1; fun(b1); D d; fun(d);B0:display(
18、)B1:display()D:display()第21页/共36页23纯虚函数在基类中不对虚函数给出有意义的实现,它只是在派生类中有具体的意义。这时基类中的虚函数只是一个入口,具体的目的地由不同的派生类中的对象决定。这个虚函数称为纯虚函数。class virtual ()=0;.;第22页/共36页24class Aprotected: int x;public:A()x =1000; virtual void print()=0; /定义纯虚函数;class B:public A /派生类private: int y;public:B() y=2000;void print()cout “y
19、=”yn;/重新定义纯虚函数;class C:public A /派生类int z;public:C()z=3000;void print()cout “z=”zprint();pa=&c; pa-print(); A a; pa=&a; pa-print( );y=2000z=3000抽象类不能定义抽象类的对象第23页/共36页251、在定义纯虚函数时,不能定义虚函数的实现部分。2、把函数名赋于0,本质上是将指向函数体的指针值赋为初值0。与定义空函数不一样,空函数的函数体为空,即调用该函数时,不执行任何动作。在没有重新定义这种纯虚函数之前,是不能调用这种函数的。第24页/共36页263、把至
20、少包含一个纯虚函数的类,称为抽象类。这种类只能作为派生类的基类,不能用来说明这种类的对象。其理由是明显的:因为虚函数没有实现部分,所以不能产生对象。但可以定义指向抽象类的指针,即指向这种基类的指针。当用这种基类指针指向其派生类的对象时,必须在派生类中重载纯虚函数,否则会产生程序的运行错误。第25页/共36页274、在以抽象类作为基类的派生类中必须有纯虚函数的实现部分,即必须有重载纯虚函数的函数体。否则,这样的派生类也是不能产生对象的。综上所述,可把纯虚函数归结为:抽象类的唯一用途是为派生类提供基类,纯虚函数的作用是作为派生类中的成员函数的基础,并实现动态多态性。第26页/共36页28虚基类多基
21、派生中的多条路径具有公共基类时,在这条路径的汇合处就会因对公共基类产生多个拷贝而产生同名函数调用的二义性。解决这个问题的办法就是把公共基类定义为虚基类,使由它派生的多条路径的汇聚处只产生一个拷贝。class Base ;class A : public Base ;class B: public Base ;class C: public A, public B ;类C中继承了两个类Base,即有两个类Base的实现部分,在调用时产生了二义性。第27页/共36页29用虚基类进行多重派生时,若虚基类没有缺省的构造函数,则在每一个派生类的构造函数中都必须有对虚基类构造函数的调用 (且首先调用)。由
22、虚基类派生出的对象初始化时,直接调用虚基类的构造函数。因此,若将一个类定义为虚基类,则一定有正确的构造函数可供所有派生类调用。第28页/共36页30class basepublic:virtual void a()couta() in basen;virtual void b()coutb() in basen;virtual void c()coutc() in basen;virtual void d()coutd() in basen;virtual void e()coute() in basen;virtual void f()coutf() in basen;class A:pub
23、lic basepublic:virtual void a()couta() in An;virtual void b()coutb() in An;virtual void f()coutf() in An;class B:public basepublic:virtual void a()couta() in Bn;virtual void b()coutb() in Bn;virtual void c()coutc() in Bn;class C:public A,public Bpublic:virtual void a()couta() in Cn;virtual void d()c
24、outa(); pa-b(); pa-c(); pa-d(); pa-e(); pa-f();将类C的地址赋值时产生歧义第29页/共36页31a( )b( )c( )d( )e( )f( )a( )b( )c( )d( )e( )f( )a( )b( )f( )a( )b( )c( )d( )e( )f( )a( )c( )a( )b( )c( )d( )e( )f( )a( )b( )f( )a( )b( )c( )d( )e( )f( )a( )c( )baseABCa( )d( )AB第30页/共36页32class basepublic:virtual void a()couta()
25、in basen;virtual void b()coutb() in basen;virtual void c()coutc() in basen;virtual void d()coutd() in basen;virtual void e()coute() in basen;virtual void f()coutf() in basen;class A:public basepublic:virtual void a()couta() in An;virtual void b()coutb() in An;virtual void f()coutf() in An;class B:pu
26、blic basepublic:virtual void a()couta() in Bn;virtual void b()coutb() in Bn;virtual void c()coutc() in Bn;class C:public A,public Bpublic:virtual void a()couta() in Cn;virtual void d()couta(); pa-b(); pa-c(); pa-d(); pa-e(); pa-f();将类C的地址赋值时产生歧义类C中有两个base,只有一个Aa() in Cb() in Ac() in based() in Ce()
27、in basef() in A为避免这种情况,将base定义为虚基类。第31页/共36页33class basepublic:virtual void a()couta() in basen;virtual void b()coutb() in basen;virtual void c()coutc() in basen;virtual void d()coutd() in basen;virtual void e()coute() in basen;virtual void f()coutf() in basen;class A:virtual public basepublic:virtu
28、al void a()couta() in An;virtual void b()coutb() in An;virtual void f()coutf() in An;class B:virtual public basepublic:virtual void a()couta() in Bn;virtual void c()coutc() in Bn;class C:public A,public Bpublic:virtual void a()couta() in Cn;virtual void d()couta(); pa-b(); pa-c(); pa-d(); pa-e(); pa
29、-f();第32页/共36页34a( )b( )c( )d( )e( )f( )a( )b( )c( )d( )e( )f( )a( )b( )f( )a( )b( )c( )d( )e( )f( )a( )c( )a( )b( )c( )d( )e( )f( )a( )b( )f( )a( )c( )baseABCa( )d( )AB第33页/共36页35class basepublic:virtual void a()couta() in basen;virtual void b()coutb() in basen;virtual void c()coutc() in basen;virtual void d()coutd() in basen;virtual void e()coute() in basen;virtual void f()
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 电力生产质量培训
- 文化遗产保护项目2025年资金申请政策解读报告
- 高压安全技术操作考试题及答案
- 文化旅游产业与创意产业融合的2025年资金申请报告
- 工商银行2025绍兴市数据分析师笔试题及答案
- 邮储银行2025三沙市笔试英文行测高频题含答案
- 2025年3D打印技术的快速成型制造研究
- 邮储银行2025淮北市秋招群面案例总结模板
- 2025年3D打印技术的创新应用领域
- 建设银行2025来宾市秋招笔试英语题专练及答案
- 2025年儿童康复学考试题库
- 《古代诗歌四首》理解性默写与训练-2023学年七年级语文上册知识梳理与能力训练
- 2025年非高危安全管理员和企业负责人习题有(含答案)
- 《高温熔融金属吊运安全规程》(AQ7011-2024)
- 2025年度乡村医生能力提升培训考试试题及答案
- 2025法拍房屋代理竞买合同范本:专业中介服务
- 医院2025年年度窗口服务优化计划
- 视网膜出血的治疗及护理
- 营销部综合事务管理办法
- 机加工车间员工技能培训
- 职业人群心理健康知识讲座:减压赋能与心理调适
评论
0/150
提交评论