“+”运算符.ppt_第1页
“+”运算符.ppt_第2页
“+”运算符.ppt_第3页
“+”运算符.ppt_第4页
“+”运算符.ppt_第5页
已阅读5页,还剩79页未读 继续免费阅读

下载本文档

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

文档简介

1、第11章 运算符重载,什么是运算符重载 运算符重载的方法 几个特殊的运算符的重载 自定义类型转换运算符 运算符重载实例,什么是运算符重载,使系统内置的运算符可以用于类类型 例如:+ 运算符能够实现2个对象间的加。例如:类A的对象a1、a2、a3,希望: a3 = a1 + a2; 即:分别把对象a1和a2的各个数据成员值对应相加,然后赋给对象a3。,问题的提出,把某些事交给系统去做,用户只要知道相加就可 扩充运算符的功能 增强了C+ 语言的可扩充性 使用户定义的类更像系统的内置类型,运算符重载的限制,不是所有的运算符都能重载 重载不能改变运算符的优先级和结合性 重载不能改变运算符的操作数个数

2、不能创建新的运算符,可以重载的运算符,+ - * / % int den; void ReductFraction(); public: Rational(int n = 0, int d = 1) num = n; den = d; Rational operator+(const Rational ,函数实现,Rational Rational:operator+(const Rational ,bool Rational:operator(const Rational ,方案二:重载成友员函数,class Rational friend Rational operator+(const

3、Rational ,函数的实现,Rational operator+(const Rational 其他函数实现略,重载后有理数类的使用,int main() Rational r1(1,6), r2(1,6), r3; r3 = r1 + r2; r1.display(); cout + ; r2.display(); cout = ; r3.display(); cout endl; r3 = r1 * r2; r1.display(); cout * ; r2.display(); cout = ; r3.display(); cout endl; return 0; ,全局函数 vs成

4、员函数,大多数运算符都可以重载成成员函数或全局函数。 赋值(=)、下标()函数调用()和成员访问(-)必须重载成成员函数。 具有赋值意义的运算符,如复合的赋值运算符以及+和-,不一定非要定义为成员函数,但最好定义为成员函数。 具有两个运算对象的运算符最好重载为全局函数,这样可以使得应用更加灵活。如果把加运算定义成全局函数,r是有理数类的对象,则2+r是一个合法的表达式。,第11章 运算符重载,什么是运算符重载 运算符重载的方法 几个特殊的运算符的重载 自定义类型转换运算符 运算符重载实例,几个特殊的运算符的重载,赋值运算符 下标运算符 函数调用运算符 +和运算符的重载 重载函数的原型设计考虑

5、输入输出运算符重载,赋值运算符,对任一类,如果用户没有自定义赋值运算符函数,那么系统为其生成一个缺省的赋值运算符函数,在对应的数据成员间赋值。 一般情况下,这个缺省的赋值运算符重载函数能满足用户的需求。但是,当类含有类型为指针的数据成员时,可能会带来一些麻烦。,对DoubleArray类对象执行array1 = array2的问题,会引起内存泄漏 使这两个数组的元素存放于同一块空间中 当这两个对象析构时,先析构的对象会释放存储数组元素的空间。而当后一个对象析构时,无法释放存放数组元素的空间,赋值运算符“=”的原型,赋值运算符只能重载成成员函数 函数原型: X 数组输入: for (i=20;

6、i arrayi; 数组输出: for (i=20; i=30; +i) cout arrayi t;,几个特殊的运算符的重载,赋值运算符 下标运算符 函数调用运算符 +和运算符的重载 重载函数的原型设计考虑 输入输出运算符重载,函数调用运算符,函数调用运算符()是一个二元运算符。它的第一个运算对象是函数名,第二个参数是形式参数表。运算的结果是函数的返回值。 一个类重载了函数调用运算符,就可以把这个类的对象当做函数来使用,函数调用运算符重载,函数调用运算符必须重载成成员函数 函数调用运算符重载函数的原型为 函数的返回值 operator() (形式参数表);,函数调用运算符重载实例,在Doub

7、leArray类增加一个功能:取数组中的一部分元素形成一个新的数组 例如,在一个下标范围为10到20的数组arr中取出下标为第12到15的元素,形成一个下标范围为2到5的数组存放在数组arr1中,可以调用 arr1 = arr(12, 15, 2)。,DoubleArray operator()(int start, int end, int lh) if (start end | start high ) cout 下标越界; exit(-1); DoubleArray tmp(lh, lh + end - start); for (int i = 0; i end - start + 1;

8、 +i) tmp.storagei = storagestart + i - low; return tmp; ,几个特殊的运算符的重载,赋值运算符 下标运算符 函数调用运算符 +和运算符的重载 重载函数的原型设计考虑 输入输出运算符重载,“+”和“-”重载,、- -:是一元操作符 这两个操作符可以是前缀,也可以是后缀。而且前缀和后缀的含义是有区别的。所以,必须有两个重载函数。 问题:两个重载函数有相同的原型 区分方法: 前缀:一元操作符。 后缀:二元操作符。,“+”和“-”重载 cont.,成员函数重载 +ob重载为:ob.operator+() ob- 重载为:ob.operator-(i

9、nt) 友元函数重载 +ob重载为:operator+(X /计数器的值 int alarm; /报警值 public: Counter(int a) value = 0; alarm = a; Counter ,类实现,Counter /返回修改前的状态 ,类的使用,int main() Counter cnt(3); /定义一个Counter类的对象,报警值为3 cnt.print(); /显示对象的当前值,此时输出为0 +cnt; cnt.print(); / 此时输出为1 (+cnt).print(); /调用前缀的+,输出2 (cnt+).print(); /调用后缀的+,当前对象的

10、value已经 /加1,报警。但输出的是2 cnt.print(); /输出值为3 return 0; ,几个特殊的运算符的重载,赋值运算符 下标运算符 函数调用运算符 +和运算符的重载 重载函数的原型设计考虑 输入输出运算符重载,重载函数的原型设计考虑,参数设计 对于任何函数的参数,如果仅需要从参数中读,而不改变它,一般用const引用来传递。 只有会修改左值参数的运算符,如赋值运算符,左值参数不是常量,所以用地址传递 返回值的类型设计 运算符的结果产生一个新值,就需要产生一个作为返回值的新对象 对于逻辑运算符,人们希望至少得到一个int或bool的返回值 所有的赋值运算符(如,=,+=等)

11、均改变左值,应该能够返回一个刚刚改变了的左值的非常量引用,值返回时的优化,在返回一个对象时,通常有两种写法。如某函数返回一个Rational类的对象,它的值为两个参数的成员对应相加。它的两种写法为 return Rational( left.num + right.num, left.den + right.den); Rational tmp; tmp.num = left.num + right.num; tmp.den = left.den + right.den; return tmp;,两种写法的比较,前者的意思是“创建一个临时对象,并返回它” 。它只调用了一次构造函数。 而后者,先

12、创建了一个对象tmp,这将调用构造函数,然后对tmp赋值,最后返回tmp。而在返回tmp时,又要创建一个临时对象,并调用拷贝构造函数用tmp对它进行初始化。在函数执行结束时,还要调用析构函数析构tmp。,几个特殊的运算符的重载,赋值运算符 下标运算符 函数调用运算符 +和运算符的重载 重载函数的原型设计考虑 输入输出运算符重载,输入输出运算符重载,输入输出运算符必须被重载成全局函数。 输出运算符的重载 输入运算符的重载,借助于流插入运算符()和流提取运算符()输入和输出用户自定义类的对象,输出重载函数的原型,ostream ,实例,ostream的结果是 1/3。,为Rational类重载输出

13、,输入输出运算符重载,输入输出运算符必须被重载成全局函数。 输出运算符的重载 输入运算符的重载,借助于流插入运算符()和流提取运算符()输入和输出用户自定义类的对象,输入重载函数的原型,istream ,实例,istream的结果是 1/3。,为Rational类重载输入,第11章 运算符重载,什么是运算符重载 运算符重载的方法 几个特殊的运算符的重载 自定义类型转换运算符 运算符重载实例,类型转换-系统预定义类型间的转换,隐式类型转换 赋值时 运算时 显式类型转换 强制转换法:(类型名)表达式 函数法:类型名(表达式),自定义类型转换运算符,类类型能否和其他的类类型或内置类型互相转换? 内置

14、类型之所以能互相转换是因为系统预先制定了转换的规则,并写好了完成转换的程序。 类类型与其它类类型或内置类型之间如何转换,编译器预先无法知道。类的设计者必须定义转换的方法。,类型转换,内置类型到类类型的转换 类类型到其它类型的转换,内置类型到类类型的转换,利用构造函数进行转换。 例如,对于Rational类的对象r,可以执行r=2。 此时,编译器隐式地调用Rational的构造函数,传给它一个参数2。构造函数将构造出一个num=2,den= 1的Rational类的对象,并将它赋给r。,explicit构造函数,任何单参数的构造函数都可以被编译器用来执行隐式转换,即把内置类型转换成对应的类类型。

15、 在某些情况下,隐式转换是不受欢迎的。 将单参数的构造函数定义为explicit,将告诉编译器不允许执行隐式转换。 如将Ratioanal类的构造函数定义成 explicit Rational(int n1 = 0, int n2 = 1) 则对于Rational类的对象r1和r2,执行 r1 = 2 + r2;编译器就会报错,类型转换,内置类型到类类型的转换 类类型到其它类型的转换,类类型到内置类型或其他类类型的转换,可以通过类型转换函数实现 类型转换函数必须重载成成员函数 类型转换函数的格式 operator 目标类型名 ( ) const return (结果为目标类型的表达式); 类型

16、转换函数的特点 无参数,无返回值 是const函数,Rational类到double的转换,转换函数的定义: operator double () const return (double(num)/den); 有了这个函数,我们可以将一个Rational类的对象r赋给一个double类型的变量x。如r的值为(1,3),经过赋值x = r后,x的值为0.333333,经过运算符重载后的Rational类,class Rational friend istream,Rational类的使用,#include #include Rational.h int main() Rational r1,

17、r2, r3, r4; double x; cout r1; cout r2; r3 = r1 + r2; cout r1 + r2 = r3 endl; r3 = r1 * r2; cout r1 * r2 = r3 endl; r4 = (r1 + r2) * r3; cout (r1 + r2) * r3的值为: r4 endl; x = 5.5 - r1; cout 5.5 - r1的值为: x endl; cout (r1 r2 ? r1 : r2) endl; return 0; ,输入r1: 1 3 输入r2: 2 6 1/3+1/3 = 2/3 1/3*1/3 = 1/9 (r

18、1 + r2) * r3的值为2/27 5.5 - r1的值为:5.16667 1/3,第11章 运算符重载,什么是运算符重载 运算符重载的方法 几个特殊的运算符的重载 自定义类型转换运算符 运算符重载实例,运算符重载实例,完善DoubleArray类,DoubleArray.h,#ifndef _array_h #define _array_h #include class DoubleArray friend ostream ,public: DoubleArray(int lh = 0, int rh = 0):low(lh), high(rh) storage = new double

19、 high - low + 1; DoubleArray(const DoubleArray #endif,DoubleArray.cpp,/文件名:DoubleArray.cpp /DoubleArray类的实现 #include #include DoubleArray.h“ DoubleArray:DoubleArray(const DoubleArray ,operator=,DoubleArray ,operator,double ,operator,ostream ,operator,istream ,operator=,bool operator=(const DoubleArr

20、ay ,operator(),DoubleArray DoubleArray:operator() (int start, int end, int lh) assert (start = low ,Main函数,int main() DoubleArray array1(20,30), array2; cin array1; cout array1 ; cout array1; array2 = array1; cout 执行 array2 = array1, array2 array2; cout array1 = array2 是 (array1 = array2) ? true : f

21、alse) endl; array225 = 0; cout 执行array25 = 0后, array1 = array2 是 (array1 = array2) ? true : false) endl; array2 = array1(22, 25, 2); cout 执行array2 = array1(22, 25, 2)后, array2 的值为: array2; return 0; ,执行结果,请输入数组元素20,30: 1 2 3 4 5 6 7 8 9 10 11 array1的内容为: 1 2 3 4 5 6 7 8 9 10 11 执行 array2 = array1,array2的内容为: 1 2 3 4 5 6 7 8 9 10 11 a

温馨提示

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

评论

0/150

提交评论