面向对象程序设计(C++)-8_第1页
面向对象程序设计(C++)-8_第2页
面向对象程序设计(C++)-8_第3页
面向对象程序设计(C++)-8_第4页
面向对象程序设计(C++)-8_第5页
已阅读5页,还剩41页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、1第八章 多态性面向对象程序设计C+2本章主要内容p多态性p运算符重载p虚函数p纯虚函数p抽象类p深度探索3多态性的概念o 多态性是面向对象程序设计的重要特征之一。o 多态性是指发出同样的消息被不同类型的对象接收时有可能导致完全不同的行为。o 多态的实现:n 函数重载n 运算符重载n 虚函数4问题举例复数的运算class Complex /复数类声明public:Complex(double r = 0.0,double i = 0.0) real = r; imag=i; void display() const; /显示复数的值private:double real;double imag

2、;运算符重载5问题举例复数的运算o 用“+”、“-”能够实现复数的加减运算吗?o 实现复数加减运算的方法 重载“+”、“-”运算符运算符重载6运算符重载的实质o运算符重载是对已有的运算符赋予多重含义o必要性nC+中预定义的运算符其运算对象只能是基本数据类型,而不适用于用户自定义类型(如类)o实现机制 将指定的运算表达式转化为对运算符函数的调用,运算对象转化为运算符函数的实参。 编译系统对重载运算符的选择,遵循函数重载的选择原则。运算符重载7规则和限制可以重载C+中除下列运算符外的所有运算符:. .* : ?:只能重载C+语言中已有的运算符,不可臆造新的。不改变原运算符的优先级和结合性。不能改变

3、操作数个数。经重载的运算符,其操作数中至少应该有一个是自定义类型。运算符重载8两种形式 重载为类的非静态成员函数 重载为非成员函数运算符重载9运算符函数o声明形式函数类型 operator 运算符(形参) . 重载为类成员函数时 参数个数=原操作数个数-1 (后置+、-除外) 重载为非成员函数时 参数个数=原操作数个数,且至少应该有一个自定义类型的形参。运算符重载10运算符成员函数的设计o 双目运算符 Bn 如果要重载 B 为类成员函数,使之能够实现表达式 oprd1 B oprd2,其中 oprd1 为A 类对象,则 B 应被重载为 A 类的成员函数,形参类型应该是 oprd2 所属的类型。

4、n 经重载后,表达式 oprd1 B oprd2 相当于 oprd1.operator B(oprd2)运算符重载11 例8-1 将“+”、“-”运算重载为复数类的成员函数。o 规则:n 实部和虚部分别相加减。实部和虚部分别相加减。o 操作数:n 两个操作数都是复数类的对象。两个操作数都是复数类的对象。运算符重载#include using namespace std;class Complex /复数类定义public: /外部接口/构造函数Complex(double r = 0.0, double i = 0.0) : real(r), imag(i) /运算符+重载成员函数Comple

5、x operator + (const Complex &c2) const; /运算符-重载成员函数Complex operator - (const Complex &c2) const;void display() const;/输出复数private:/私有数据成员double real;/复数实部double imag;/复数虚部;12/重载运算符函数实现Complex Complex:operator + (const Complex &c2) const /创建一个临时无名对象作为返回值return Complex(real + c2.real, imag

6、 + c2.imag); /重载运算符函数实现Complex Complex:operator - (const Complex &c2) const /创建一个临时无名对象作为返回值return Complex(real - c2.real, imag - c2.imag); 13void Complex:display() const cout ( real , imag ) endl;int main() /主函数Complex c1(5, 4), c2(2, 10), c3; /定义复数类的对象cout c1 = ; c1.display();cout c2 = ; c2.di

7、splay();c3 = c1-c2;/使用重载运算符完成复数减法cout c3 = c1-c2 = ; c3.display();c3 = c1 + c2;/使用重载运算符完成复数加法cout c3 = c1 + c2 = ; c3.display();return 0;14程序输出的结果为:c1 = (5, 4)c2 = (2, 10)c3 = c1 - c2 = (3, -6)c3 = c1 + c2 = (7, 14)15运算符成员函数的设计o 前置单目运算符 Un 如果要重载 U 为类成员函数,使之能够实现表达式 U oprd,其中 oprd 为A类对象,则 U 应被重载为 A 类的

8、成员函数,无形参。n 经重载后,表达式 U oprd 相当于 oprd.operator U()运算符重载16运算符成员函数的设计o 后置单目运算符 +和-n 如果要重载 +或-为类成员函数,使之能够实现表达式 oprd+ 或 oprd- ,其中 oprd 为A类对象,则 +或- 应被重载为 A 类的成员函数,且具有一个 int 类型形参。n 经重载后,表达式 oprd+ 相当于 oprd.operator +(0)运算符重载17例8-2o 运算符前置+和后置+重载为时钟类的成员函数。o 前置单目运算符,重载函数没有形参,对于后置单目运算符,重载函数需要有一个整型形参。o 操作数是时钟类的对象

9、。o 实现时间增加1秒钟。运算符重载#include using namespace std;class Clock /时钟类声明定义public:/外部接口Clock(int hour = 0, int minute = 0, int second = 0);void showTime() const;Clock& operator + (); /前置单目运算符重载Clock operator + (int); /后置单目运算符重载private:/私有数据成员int hour, minute, second;18/前置单目运算符重载函数Clock & Clock:opera

10、tor + () second+;if (second = 60) second -= 60;minute+;if (minute = 60) minute -= 60;hour = (hour + 1) % 24;return *this;19/后置单目运算符重载Clock Clock:operator + (int) /注意形参表中的整型参数Clock old = *this;+(*this);/调用前置“+”运算符return old;20/其它成员函数的实现略int main() Clock myClock(23, 59, 59);cout First time output: ;my

11、Clock.showTime();cout Show myClock+: ;(myClock+).showTime();cout Show +myClock: ;(+myClock).showTime();return 0;21程序运行结果为:First time output: 23:59:59Show myClock+: 23:59:59Show +myClock: 0:0:122运算符非成员函数的设计o 函数的形参代表依自左至右次序排列的各操作数。o 后置单目运算符 +和-的重载函数,形参列表中要增加一个int,但不必写形参名。o 如果在运算符的重载函数中需要操作某类对象的私有成员,可以

12、将此函数声明为该类的友元。运算符重载23运算符非成员函数的设计o 双目运算符 B重载后,表达式oprd1 B oprd2 等同于operator B(oprd1,oprd2 )o 前置单目运算符 B重载后,表达式 B oprd 等同于operator B(oprd )o 后置单目运算符 +和-重载后,表达式 oprd B 等同于operator B(oprd,0 )运算符重载24例8-3o 将+、-(双目)重载为非成员函数,并将其声明为复数类的友元,两个操作数都是复数类的常引用。o 将(双目)重载为非成员函数,并将其声明为复数类的友元,它的左操作数是std:ostream引用,右操作数为复数类

13、的常引用,返回std:ostream引用,用以支持下面形式的输出:cout a b;该输出调用的是:operator (operator (cout, a), b);运算符重载#include using namespace std;class Complex /复数类定义public: /外部接口 /构造函数Complex(double r = 0.0, double i = 0.0) : real(r), imag(i) /运算符+重载friend Complex operator + (const Complex &c1, const Complex &c2); /运算符

14、-重载friend Complex operator - (const Complex &c1, const Complex &c2); /运算符重载friend ostream & operator (ostream &out, const Complex &c); private: /私有数据成员double real; /复数实部double imag;/复数虚部;25Complex operator + (const Complex &c1, const Complex &c2) return Complex(c1.real +

15、c2.real, c1.imag + c2.imag); Complex operator - (const Complex &c1, const Complex &c2) return Complex(c1.real - c2.real, c1.imag - c2.imag); ostream & operator (ostream &out, const Complex &c) out ( c.real , c.imag );return out;2627静态绑定与动态绑定o 绑定n 程序自身彼此关联的过程,确定程序中的操作调用与执行该操作的代码间的

16、关系。o 静态绑定n 绑定过程出现在编译阶段,用对象名或者类名来限定要调用的函数。o 动态绑定n 绑定过程工作在程序运行时执行,在程序运行时才确定将要调用的函数。#includeusing namespace std;class Point public:Point(double x, double y) : x(x), y(y) double area() const return 0.0; private:double x, y;class Rectangle: public Point public:Rectangle(double x, double y, double w, doub

17、le h);double area() const return w * h; private:double w, h;静态绑定例28Rectangle:Rectangle(double x, double y, double w, double h) :Point(x, y), w(w), h(h) void fun(const Point &s) cout Area = s.area() endl;int main() Rectangle rec(3.0, 5.2, 15.0, 25.0);fun(rec);return 0;运行结果:Area = 029#includeusing

18、 namespace std;class Point public:Point(double x, double y) : x(x), y(y) virtual double area() const return 0.0; private:double x, y;class Rectangle:public Point public:Rectangle(double x, double y, double w, double h);virtual double area() const return w * h; private:double w, h;/其他函数同上例动态绑定例 30voi

19、d fun(const Point &s) cout Area = s.area() endl;int main() Rectangle rec(3.0, 5.2, 15.0, 25.0);fun(rec);return 0;运行结果:Area = 3753132虚函数虚函数是动态绑定的基础。是非静态的成员函数。在类的声明中,在函数原型之前写virtual。virtual 只用来说明类声明中的原型,不能用在函数实现时。具有继承性,基类中声明了虚函数,派生类中无论是否说明,同原型函数都自动为虚函数。o本质:不是重载声明而是派生类覆盖基类的同名虚函数。o调用方式:通过基类指针或引用,执行时

20、会根据指针指向的对象的类,决定调用哪个函数。虚 函 数33例 8-4#include using namespace std; class Base1 /基类Base1定义public:virtual void display() const; /虚函数;void Base1:display() const cout Base1:display() endl;class Base2: public Base1 /公有派生类Base2定义public:void display() const;/覆盖基类的虚函数;void Base2:display() const cout Base2:disp

21、lay() endl;虚 函 数/公有派生类Derived定义class Derived: public Base2 public:void display() const; /覆盖基类的虚函数;void Derived:display() const cout Derived:display() display(); /对象指针-成员名34int main() /主函数Base1 base1;/定义Base1类对象Base2 base2;/定义Base2类对象Derived derived; /定义Derived类对象fun(&base1);/用Base1对象的指针调用fun函数fu

22、n(&base2);/用Base2对象的指针调用fun函数fun(&derived);/用Derived对象的指针调用fun函数return 0;运行结果:Base1:display()Base2:display()Derived:display()3536虚析构函数为什么需要虚析构函数? 可能通过基类指针删除派生类对象; 如果你打算允许其他人通过基类指针调用派生类对象的析构函数,就需要让基类的析构函数成为虚函数。虚 函 数37抽象类带有纯虚函数的类称为抽象类:class 类名 virtual 类型 函数名(参数表)=0; /纯虚函数 .纯虚函数与抽象类38抽象类o作用n抽象类

23、为抽象和设计的目的而声明,将有关的数据和行为组织在一个继承层次结构中,保证派生类具有要求的行为。n对于暂时无法实现的函数,可以声明为纯虚函数,留给派生类去实现。o注意 抽象类只能作为基类来使用。 不能声明抽象类的对象。 构造函数不能是虚函数,析构函数可以是虚函数。纯虚函数与抽象类39例 8-5#include using namespace std;class Base1 /基类Base1定义public:virtual void display() const = 0; /纯虚函数;class Base2: public Base1 /公有派生类Base2定义public:void disp

24、lay() const /覆盖基类的虚函数cout Base2:display() endl;纯虚函数与抽象类class Derived: public Base2 /公有派生类Derived定义public:void display() const /覆盖基类的虚函数cout Derived:display() display();/对象指针-成员名int main() /主函数Base2 base2;/定义Base2类对象Derived derived; /定义Derived类对象fun(&base2);/用Base2对象的指针调用fun函数fun(&derived); /

25、用Derived对象的指针调用fun函数return 0;运行结果:Base2:display()Derived:display()40多态类型与非多态类型o 多态类型与非多态类型n 有虚函数的类类型称为多态类型n 其它类型皆为非多态类型o 二者的差异n 语言层面的差异o 多态类型支持运行时类型识别o 多态类型对象占用额外的空间n 设计原则上的差异41深 度 探 索设计原则o 多态类型n 多态类型的析构函数一般应为虚函数o 非多态类型n 非多态类型不宜作为公共基类o 由于没有利用动态多态性,一般可以用组合,而无需用共有继承;o 如果继承,则由于析构函数不是虚函数,删除对象时所执行操作与指针类型有关,易引起混乱。n 把不需被继承的类型设定为非多态类型o 由于成员函数都是静态绑定,调用速度较快;o 对象占用空间较小。42深 度 探 索运行时类型识别o 运行时类型识别n 允许在运行时通过基类指针(或引用)辨别对象所属的具体派生类;n 只对多态类型适用;n 比虚函数动态绑定的开销更大,因此应仅对虚函数无

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论