1、单击此处编辑母版标题样式,单击此处编辑母版文本样式,第二级,第三级,第四级,第五级,*,第,7,章 运算符重载,7.1,运算符重载概述,7.2,运算符重载的实现,7.3,单目运算符重载,7.4,双目运算符重载,7.5 -,运算符重载,7.6,赋值运算符重载,7.7,下标运算符重载,7.8,运算符,new,与,delete,重载,7.9,逗号运算符重载,7.10,类型转换运算符重载,课后思考题,C+,语言允许程序员重新定义已有的运算符,使其能按用户的要求完成一些特定的操作,这就是所谓的,运算符重载,(operator overload),。运算符重载与函数重载相似,其目的是设置某一运算符,让它具
2、有另一种功能,尽管此运算符在原先,C+,语言中代表另一种含义,但他们彼此之间并不冲突。,C+,会根据运算符的位置辨别应使用哪一种功能进行运算。,7.1,运算符重载概述,运算符重载,是对系统提供的运算符赋予多重含义,同一个运算符作用于不同类型的数据导致不同类型的行为。,运算符重载的实质就是函数重载。,在实现过程中,首先把指定的运算表达式转化为对运算符函数的调用,运算对象转化为运算符函数的实参,然后根据实参的类型来确定需要调用的函数,这个过程是在编译过程中完成的。,C+,可以重载的运算符如表,7-1,所示;,C+,不能被重载的运算符如表,7-2,所示。,表,7-1,:,C+,可以重载的运算符,!,
3、/,%,=,=,!=,|,&,+=,=,*,=,/=,%=,&=,=,|=,=,-*,-,(),=,+,-,new,delete,表,7-2,:,C+,不可以重载的运算符,.,.*,:,?:,#,#,运算符重载的规则如下:,(,1,),C+,中的运算符,除了少数几个以外,,几乎全部可以重载,而且只能重载已有的这些运算符。,(,2,),重载之后运算符的优先级和结合性都不会改变,。,(,3,)运算符重载是针对新类型数据的实际需要,对原有运算符进行适当的改造。一般来讲,重载的功能应当与原有,功能类似,。,(,4,),运算符重载不能改变原运算符的操作对象个数,同时至少要有一个操作对象是
4、自定义类型。,7.2,运算符重载的实现,运算符的重载形式有两种:重载为类的成员函数和重载为类的友元函数。,1.,运算符重载为类的成员函数的语法形式如下:,operator,(,),;,friend operator,(,),;,其中,:,指定了重载运算符的返回值类型,;,operator,是定义运算符重载函数的关键词,;,给定了要重载的运算符名称,是,C+,中可重载的运算符,;,中给出重载运算符所需要的参数和类型,;,对于运算符重载为友元函数的情况,还要在函数类型说明之前使用,friend,关键词来说明。,注意事项:,当运算符重载为类的成员函数时,函数的参数个数比原来的运算数个数要少一个(后缀
5、除外);当重载为类的友元函数时,参数个数与原运算数的个数相同。,单目运算符最好重载为成员函数;,双目运算符则最好重载为友元函数;,运算符重载的含义必须清楚;,运算符重载不能有二义性;,7.3,单目运算符重载,类的单目运算符可重载为一个没有参数的非静态成员函数或者带有一个参数的非成员函数,参数必须是用户字定义类型的对象或者是对该对象的引用。,在,C+,中,单目运算符有,+,和,-,,它们是变量自动增,1,和自动减,1,的运算符。在类中可以对这两个单目运算符进行重载。,如同“,+”,运算符有前缀、后缀两种使用形式,“,+”,和“,-”,重载运算符也有前缀和后缀两种运算符重载形式,以“
6、重载运算符为例,其语法格式如下:,operator+,();,/,前缀运算,operator+,(,int,),;,/,后缀运算,使用前缀运算符的语法格式如下:,+,;,使用运算符前缀时,对对象(操作数)进行增量修改,然后再返回该对象。所以前缀运算符操作时,参数与返回的是同一个对象。这与基本数据类型的运算符前缀类似,返回的也是左值。,使用后缀运算符的语法格式如下:,+,;,使用运算符后缀时,必须再增量之前返回原有的对象值。为此,需要创建一个临时对象,存放原有的对象,以便对操作数(对象)进行增量修改时,保存最初的值。运算符后缀操作时返回的时原有对象值,不是原有对象,原有对象已经被增量修改
7、所以,返回的应该是存放原有对象值的临时对象。,(见例,1,:,+,运算符重载),7.4,双目运算符重载,对于双目运算符,一个运算数是对象本身的数据,由,this,指针给出,另一个运算数则需要通过运算符重载函数的参数表来传递。下面分别介绍这两种情况。,将双目运算符,B,重载为类的成员函数:,使之能够实现表达式“,oprd1 B oprd2”,,,其中,oprd1,为,A,类的对象,则应当把,B,重载为,A,类的成员函数,该函数只有一个形参,形参的类型是,oprd2,所属的类型。经过重载之后,表达式,oprd1 B oprd2,就相当于函数调用“,oprd1.operator B,(,oprd2
8、将双目运算符,B,重载为类的友元函数:,这样,它就可以自由地访问该类的任何数据成员。这时,运算符所需要的运算数都需要通过函数的形参表来传递,在参数表中形参从左到右的顺序就是运算符运算数的顺序。,(见例,2,:双目运算符重载),7.5 -,运算符重载,“,-”,运算符是成员访问运算符,这种一元的运算符,只能被重载为成员函数,,所以也决定了它不能定义任何参数。,1.,成员访问运算符“,-”,函数重载的一般形式为:,class_name,*,class_name:operator,-,();,2.,成员访问运算符的调用形式是:,对象,-,成员;,/,与对象指针调用成员比较,注意:,“,-
9、成员重载运算符不能是静态成员函数;,通过对象,-,成员;实现访问的成员应该是,public,型;,(,/,例,10,:“,-”,运算符重载为成员函数),7.6,赋值运算符重载,在,C+,中有两种类型的赋值运算符:一类是“,+=”,和“,-=”,等先计算后赋值的运算符,另一类是“,=”,即直接赋值的运算符。下面分别进行讨论。,1.,运算符“,+=”,和“,-=”,的重载,对于标准数据类型,“,+=”,和“,-=”,的作用是将一个数据与另一个数据进行加法或减法运算后再将结果回送给赋值号左边的变量中。对它们重载后,使其实现其他相关的功能。,(见例,4,:复合赋值运算符重载),2.,运算符“,=”
10、的重载,赋值运算符“,=”,的原有含义是将赋值号右边表达式的结果拷贝给赋值号左边的变量,通过运算符“,=”,的重载将赋值号右边对象的私有数据依次拷贝到赋值号左边对象的私有数据中。在正常情况下,系统会为每一个类自动生成一个默认的完成上述功能的赋值运算符,当然,这种赋值只限于由一个类类型说明的对象之间赋值。,如果一个类包含指针成员,采用这种默认的按成员赋值,那么当这些成员撤消后,内存的使用将变得不可靠。,(见例,5,:赋值运算符“,=”,的重载),7.7,下标运算符重载,下标运算符“,”,通常用于在数组中标识数组元素的位置,通过下标运算符重载可以实现数组数据的赋值和取值。,下标运算符实质是求地址
11、运算。,下标运算符重载函数只能作为类的成员函数,不能作为类的友元函数。,下标运算符“,”,函数重载的一般形式为:,type class_name:operator,(,int,arg,),;,其中,arg,为该重载函数的参数。重载了的下标运算符只能且必须带一个参数,该参数给出下标的值。重载函数,operator,的返回值类型,type,是引用类型。,(见例,6,:下标运算符重载),7.8,运算符,new,与,delete,重载,C+,提供了,new,与,delete,两个运算符用于内存管理,但有些情况下用户需要自己管理内存,为自己所定义的类体系建立一种新的动态内存管理算法,以克服,new,与,
12、delete,的不足。这就要重载运算符,new,与,delete,,,使其按照要求完成对内存的管理。,注意:,new,和,delete,只能被重载为类的成员函数,不能重载为友元。,重载了的,new,和,delete,均默认为类的静态成员函数。,运算符,new,重载的一般形式为:,void*class_name:operator new,(,size_t,,,);,new,重载应返回一个无值型的指针,且至少有一个类型为,size_t,的参数。若该重载带有多于一个的参数,则其第一个参数的类型必须为,size_t,。,运算符,delete,重载的一般形式为:,void*class_name:oper
13、ator delete,(,void*,,,);,delete,重载应返回一个无值型的指针,且至少有一个类型为无值型指针的参数。该重载最多可以带有两个参数,若有第二个参数,则其第二个参数的类型必须为,size_t,。,(见例,7,:运算符,new,与,delete,重载),7.9,逗号运算符重载,逗号运算符是双目运算符,和其他运算符一样,也可以通过重载逗号运算符来完成期望完成的工作。逗号运算符构成的表达式为“左运算数,右运算数”,该表达式返回右运算数的值。如果用类的成员函数来重载逗号运算符,则只带一个右运算数,而左运算数由指针,this,提供。,(见例,8,:逗号运算符重载),7.10,类型转
14、换运算符重载,类型转换运算符重载函数的格式如下:,operator,(),;,与以前的重载运算符函数不同的是,,类型转换运算符重载函数没有返回类型,因为,就代表了它的返回类型,而且也没有任何参数。,在调用过程中要带一个对象实参。,类型转换运算符的实质将对象转换成类型名规定的类型。,转换时的形式就像强制转换一样。如果没有转换运算符定义,直接用强制转换是不行的,因为强制转换只能对标准数据类型进行操作,对类类型的操作是没有定义的。,类型转换运算符重载,的,缺点,是无法定义其类对象运算符操作的真正含义,因为只能进行相应对象成员数据和一般数据变量的转换操作。,(见例,9,:类型转换运算符重载),本章结束,课后思考题(编程验证),将“,-”,运算符重载为成员函数,将“,”,和“,”,运算符重载,测试类类型转换运算符重载。如,Circle(,圆类,),与,Point,(点类)的转换。,测试运算符,new,和,delete,的重载,要求参数多于,1,个。,






