《chap8C课件清华大学郑莉.pptx》由会员分享,可在线阅读,更多相关《chap8C课件清华大学郑莉.pptx(42页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、1本章主要内容多态性运算符重载虚函数纯虚函数抽象类第1页/共42页2多态性的概念多态性是面向对象程序设计的重要特征之一。多态性是指发出同样的消息被不同类型的对象接收时有可能导致完全不同的行为。多态的实现:函数重载运算符重载虚函数第2页/共42页3问题举例复数的运算class complex/复数类声明public:complex(double r=0.0,double i=0.0)/构造函数 real=r;imag=i;void display();/显示复数的值private:double real;double imag;运算符重载第3页/共42页4问题举例复数的运算用“+”、“-”能够实
2、现复数的加减运算吗?实现复数加减运算的方法 重载“+”、“-”运算符运算符重载第4页/共42页5运算符重载的实质运算符重载是对已有的运算符赋予多重含义必要性C+中预定义的运算符其运算对象只能是基本数据类型,而不适用于用户自定义类型(如类)实现机制将指定的运算表达式转化为对运算符函数的调用,运算对象转化为运算符函数的实参。编译系统对重载运算符的选择,遵循函数重载的选择原则。运算符重载第5页/共42页6运算符重载规则和限制可以重载C+中除下列运算符外的所有运算符:.*:?:只能重载C+语言中已有的运算符,不可臆造新的。不改变原运算符的优先级和结合性。不能改变操作数个数。经重载的运算符,其操作数中至
3、少应该有一个是自定义类型。第6页/共42页7两种形式重载为类成员函数。重载为友元函数。运算符重载第7页/共42页8运算符函数声明形式函数类型 operator 运算符(形参).重载为类成员函数时 参数个数=原操作数个数-1(后置+、-除外)重载为友元函数时 参数个数=原操作数个数,且至少应该有一个自定义类型的形参。运算符重载第8页/共42页9运算符成员函数的设计双目运算符 B如果要重载 B 为类成员函数,使之能够实现表达式 oprd1 B oprd2,其中 oprd1 为A 类对象,则 B 应被重载为 A 类的成员函数,形参类型应该是 oprd2 所属的类型。经重载后,表达式 oprd1 B
4、oprd2 相当于 oprd1.operator B(oprd2)运算符重载第9页/共42页10运算符重载 例 8.1 将“+”、“-”运算重载为复数类的成员函数。规则:实部和虚部分别相加减。操作数:两个操作数都是复数类的对象。第10页/共42页#includeusing namespace std;class complex/复数类声明public:/外部接口complex(double r=0.0,double i=0.0)real=r;imag=i;/构造函数complex operator+(complex c2);/+重载为成员函数complex operator-(complex
5、c2);/-重载为成员函数void display();/输出复数private:/私有数据成员double real;/复数实部double imag;/复数虚部;11第11页/共42页complex complex:operator+(complex c2)/重载函数实现complex c;c.real=c2.real+real;c.imag=c2.imag+imag;return complex(c.real,c.imag);12第12页/共42页complex complex:operator-(complex c2)/重载函数实现complex c;c.real=real-c2.re
6、al;c.imag=imag-c2.imag;return complex(c.real,c.imag);13第13页/共42页void complex:display()cout(real,imag)endl;int main()/主函数 complex c1(5,4),c2(2,10),c3;/声明复数类的对象coutc1=;c1.display();coutc2=;c2.display();c3=c1-c2;/使用重载运算符完成复数减法coutc3=c1-c2=;c3.display();c3=c1+c2;/使用重载运算符完成复数加法coutc3=c1+c2=;c3.display();
7、14第14页/共42页程序输出的结果为:c1=(5,4)c2=(2,10)c3=c1-c2=(3,-6)c3=c1+c2=(7,14)15第15页/共42页16运算符成员函数的设计前置单目运算符 U如果要重载 U 为类成员函数,使之能够实现表达式 U oprd,其中 oprd 为A类对象,则 U 应被重载为 A 类的成员函数,无形参。经重载后,表达式 U oprd 相当于 oprd.operator U()运算符重载第16页/共42页17运算符成员函数的设计后置单目运算符+和-如果要重载+或-为类成员函数,使之能够实现表达式 oprd+或 oprd-,其中 oprd 为A类对象,则+或-应被重
8、载为 A 类的成员函数,且具有一个 int 类型形参。经重载后,表达式 oprd+相当于 oprd.operator+(0)运算符重载第17页/共42页18例8.2运算符前置+和后置+重载为时钟类的成员函数。前置单目运算符,重载函数没有形参,对于后置单目运算符,重载函数需要有一个整型形参。操作数是时钟类的对象。实现时间增加1秒钟。运算符重载第18页/共42页/8_2.cpp#includeusing namespace std;class Clock/时钟类声明 public:/外部接口 Clock(int NewH=0,int NewM=0,int NewS=0);void ShowTime
9、();Clock&operator+();/前置单目运算符重载 Clock operator+(int);/后置单目运算符重载 private:/私有数据成员 int Hour,Minute,Second;19第19页/共42页Clock&Clock:operator+()/前置单目运算符重载函数 Second+;if(Second=60)Second=Second-60;Minute+;if(Minute=60)Minute=Minute-60;Hour+;Hour=Hour%24;return*this;20第20页/共42页/后置单目运算符重载Clock Clock:operator+(
10、int)/注意形参表中的整型参数 Clock old=*this;+(*this);return old;21第21页/共42页/其它成员函数的实现略int main()Clock myClock(23,59,59);coutFirst time output:;myClock.ShowTime();coutShow myClock+:;(myClock+).ShowTime();coutShow+myClock:;(+myClock).ShowTime();22第22页/共42页程序运行结果为:First time output:23:59:59Show myClock+:23:59:59S
11、how+myClock:0:0:123第23页/共42页24运算符友元函数的设计如果需要重载一个运算符,使之能够用于操作某类对象的私有成员,可以此将运算符重载为该类的友元函数。函数的形参代表依自左至右次序排列的各操作数。后置单目运算符+和-的重载函数,形参列表中要增加一个int,但不必写形参名。运算符重载第24页/共42页25运算符友元函数的设计双目运算符 B重载后,表达式oprd1 B oprd2 等同于operator B(oprd1,oprd2)前置单目运算符 B重载后,表达式 B oprd 等同于operator B(oprd)后置单目运算符+和-重载后,表达式 oprd B 等同于o
12、perator B(oprd,0)运算符重载第25页/共42页26例8-3将+、-(双目)重载为复数类的友元函数。两个操作数都是复数类的对象。运算符重载第26页/共42页#includeusing namespace std;class complex/复数类声明public:/外部接口complex(double r=0.0,double i=0.0)real=r;imag=i;/构造函数friend complex operator+(complex c1,complex c2);/运算符+重载为友元函数friend complex operator-(complex c1,complex
13、 c2);/运算符-重载为友元函数void display();/显示复数的值private:/私有数据成员double real;double imag;27第27页/共42页complex operator+(complex c1,complex c2)/运算符重载友元函数实现 return complex(c2.real+c1.real,c2.imag+c1.imag);complex operator-(complex c1,complex c2)/运算符重载友元函数实现 return complex(c1.real-c2.real,c1.imag-c2.imag);/其它函数和主函数
14、同例8.128第28页/共42页29静态绑定与动态绑定绑定程序自身彼此关联的过程,确定程序中的操作调用与执行该操作的代码间的关系。静态绑定(静态联编)联编工作出现在编译阶段,用对象名或者类名来限定要调用的函数。动态绑定联编工作在程序运行时执行,在程序运行时才确定将要调用的函数。第29页/共42页34虚函数虚函数是动态绑定的基础。是非静态的成员函数。在类的声明中,在函数原型之前写virtual。virtual 只用来说明类声明中的原型,不能用在函数实现时。具有继承性,基类中声明了虚函数,派生类中无论是否说明,同原型函数都自动为虚函数。本质:不是重载声明而是覆盖。调用方式:通过基类指针或引用,执行
15、时会根据指针指向的对象的类,决定调用哪个函数。虚 函 数第34页/共42页35例 8.4#include using namespace std;class B0/基类B0声明public:/外部接口virtual void display()/虚成员函数 coutB0:display()endl;class B1:public B0/公有派生 public:void display()coutB1:display()endl;class D1:public B1/公有派生 public:void display()coutD1:display()display();int main()/主函
16、数B0 b0,*p;/声明基类对象和指针B1 b1;/声明派生类对象D1 d1;/声明派生类对象p=&b0;fun(p);/调用基类B0函数成员p=&b1;fun(p);/调用派生类B1函数成员p=&d1;fun(p);/调用派生类D1函数成员运行结果:B0:display()B1:display()D1:display()36第36页/共42页37虚析构函数何时需要虚析构函数?当你可能通过基类指针删除派生类对象时如果你打算允许其他人通过基类指针调用对象的析构函数(通过delete这样做是正常的),并且被析构的对象是有重要的析构函数的派生类的对象,就需要让基类的析构函数成为虚拟的。虚 函 数第
17、37页/共42页38抽象类带有纯虚函数的类称为抽象类:class 类名 virtual 类型 函数名(参数表)=0;/纯虚函数 .纯虚函数与抽象类第38页/共42页39抽象类纯虚函数与抽象类作用抽象类为抽象和设计的目的而声明,将有关的数据和行为组织在一个继承层次结构中,保证派生类具有要求的行为。对于暂时无法实现的函数,可以声明为纯虚函数,留给派生类去实现。注意抽象类只能作为基类来使用。不能声明抽象类的对象。构造函数不能是虚函数,析构函数可以是虚函数。第39页/共42页40例 8.5纯虚函数与抽象类#include using namespace std;class B0 /抽象基类B0声明 p
18、ublic:/外部接口virtual void display()=0;/纯虚函数成员;class B1:public B0 /公有派生 public:void display()coutB1:display()endl;/虚成员函数;class D1:public B1 /公有派生 public:void display()coutD1:display()display();int main()/主函数B0*p;/声明抽象基类指针B1 b1;/声明派生类对象D1 d1;/声明派生类对象p=&b1;fun(p);/调用派生类B1函数成员p=&d1;fun(p);/调用派生类D1函数成员运行结果:B1:display()D1:display()41第41页/共42页42感谢您的观赏!第42页/共42页