《c++程序设计》课件第7章 运算符重载_第1页
《c++程序设计》课件第7章 运算符重载_第2页
《c++程序设计》课件第7章 运算符重载_第3页
《c++程序设计》课件第7章 运算符重载_第4页
《c++程序设计》课件第7章 运算符重载_第5页
已阅读5页,还剩104页未读 继续免费阅读

下载本文档

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

文档简介

第7章运算符重载2第7章运算符重载本章学习要点为什么要进行运算符重载运算符重载的方法重载运算符的规则运算符重载函数作为类成员函数和友元函数重载双目运算符重载单目运算符重载流插入运算符和流提取运算符不同类型数据间的转换3第7章运算符重载函数重载,就是赋给同一个函数名多个含义。运算符重载是指同一个运算符可以施加于不同类型的操作数上面。4classMoney{public:Money(inty=0,intj=0,intf=0){yuan=y;jiao=j;fen=f;optimize();}voidDisplay(string);private:intyuan,jiao,fen;voidOptimize();//优化函数};voidMoney::Optimize(){if(fen>=10){jiao++;fen-=10;}if(jiao>=10){yuan++;jiao-=10;}}voidMoney::Display(stringstr){cout<<str<<"="<<yuan<<"."<<jiao<<fen<<"¥"<<endl;}Moneycost1(10,3,5);Moneycost2(5,8,2),total;total=cost1+cost2;构造函数5classMoney{public:Money(inty=0,intj=0,intf=0){yuan=y;jiao=j;fen=f;optimize();}voidDisplay(string);

friend

MoneyMoneyAdd(Money&c1,Money&c2);private:intyuan,jiao,fen;voidOptimize();};voidMoney::Optimize(){if(fen>=10){jiao++;fen-=10;}if(jiao>=10){yuan++;jiao-=10;}}voidMoney::Display(stringstr){cout<<str<<"="<<yuan<<"."<<jiao<<fen<<"¥"<<endl;}MoneyMoneyAdd(Money&c1,Money&c2){

returnMoney(c1.yuan+c2.yuan,c1.jiao+c2.jiao,c1.fen+c2.fen);}intmain(){Moneycost1(300,5,6),cost2(105,7,6),total1;total1=MonetAdd(cost1,cost2);total1.Display("total1=cost1+cost2");return0;}6classMoney{public:Money(inty=0,intj=0,intf=0){yuan=y;jiao=j;fen=f;optimize();}voidDisplay(string);MoneyMoneyAdd(Money&c2);private:intyuan,jiao,fen;voidOptimize();};voidMoney::Optimize(){if(fen>=10){jiao++;fen-=10;}if(jiao>=10){yuan++;jiao-=10;}}voidMoney::Display(stringstr){cout<<str<<"="<<yuan<<"."<<jiao<<fen<<"¥"<<endl;}MoneyMoney::MoneyAdd(Money&c2){returnMoney(yuan+c2.yuan,jiao+c2.jiao,fen+c2.fen);}intmain(){Moneycost1(300,5,6),cost2(105,7,6),total1;total1=cost1.MonetAdd(cost2);total1.Display("total1=cost1+cost2");return0;}77.1为什么要进行运算符重载但我们希望在程序中直接用运算符“+”对Money类的对象进行相加运算。

total1=cost1+cost2;对运算符“+”进行重载以后,就可以这样书写了。87.1为什么要进行运算符重载有了针对自定义类型数据的运算符重载,不仅使我们编程时感到十分方便,而且写出的表达式与数学表达式很相似,符合人们的习惯。97.2运算符重载的方法重载运算符的函数的一般格式如下:函数类型operator运算符名称(形参列表){对运算符的重载处理}函数名函数名是由operator和运算符组成,如operator+意思是对运算符“+”重载。重载运算符的函数可以是类的成员函数,也可以是类的友元函数,还可以是既非类的成员函数也非类的友元函数的普通函数10【例7-1】对“+”运算符进行重载来实现两个Money类对象的加法运算。7.2运算符重载的方法11#include<iostream>#include<string>usingnamespacestd;classMoney{public:Money(inty=0,intj=0,intf=0);Moneyoperator+(Money&);voidDisplay(string);private:intyuan,jiao,fen;voidOptimize();};voidMoney::Optimize(){if(fen>=10){jiao++;fen-=10;}if(jiao>=10){yuan++;jiao-=10;}}Money::Money(inty,intj,intf){yuan=y;jiao=j;fen=f;Optimize();}MoneyMoney::operator+(Money&c2){returnMoney(yuan+c2.yuan,jiao+c2.jiao,fen+c2.fen);}voidMoney::Display(stringstr){cout<<str<<"="<<yuan<<"."<<jiao<<fen<<"¥"<<endl;}函数名intmain(){Moneycost1(300,5,6),cost2(105,7,6),total1,total2;total1=cost1+cost2;total2=cost1.operator+(cost2);total1.Display("total1=cost1+cost2");total2.Display("total2=cost1+cost2");return0;}12运算符重载函数可以是:⑴类的成员函数

可以通过this指针自由地访问本类的数据成员,少写一个函数参数。7.2运算符重载的方法成员函数定义:MoneyMoney::operator+(Money&c2){returnMoney(yuan+c2.yuan,jiao+c2.jiao,fen+c2.fen);}ComplexComplex::operator+(Complex&c2){returnComplex(real+c2.real,imag+c2.imag);}调用:

total1=cost1+cost2;表达式cost1+cost2中第一个参数cost1是Money类对象,运算符函数返回值的类型也是Money,这是正确的。如果cost1不是Money类的对象,它就无法通过隐式this指针访问Money类的成员了。如果函数返回值不是Money类复数,显然这种运算是没有实际意义的。7.2运算符重载的方法14⑵类的友元函数双目运算符重载为友元函数时,在函数的形参表列中必须有两个参数。声明:friendMoneyoperator+(Money&c1,Money&c2);友元函数定义:Moneyoperator+(Money&c1,Money&c2){returnComplex(c1.yuan+c2.yuan,c1.jiao+c2.jiao,c1.fen+c2.fen);}调用:total1=cost1+cost2;形参的顺序任意,不要求第一个参数必须为类对象。要求运算符左侧的操作数与第一个参数对应,运算符右侧的操作数与第二个参数对应。7.2运算符重载的方法157.3重载运算符的规则(1)C++不允许用户自己定义新的运算符,只能对C++语言中已有的运算符进行重载。例如,虽然在某些程序设计语言中用双字符“**”作为求幂运算符,但是在使用C++进行程序设计时,不能将“**”作为运算符进行重载,因为“**”不是C++语言的合法运算符。(2)运算符重载针对新类型数据的实际需要,对原有运算符进行适当的改造。一般来讲,重载的功能应当与原有功能相类似。7.3重载运算符的规则(3)C++允许重载的运算符包括C++中几乎所有的运算符。具体规定如表7-1所示。表7-1C++允许重载的运算符运算符名称具体运算符算术运算符+(加),-(减),*(乘),/(除),%(取模),++(自增),--(自减)位操作运算符&(按位与),~(按位取反),^(按位异或),|(按位或),<<(左移),>>(右移)逻辑运算符!(逻辑非),&&(逻辑与),||(逻辑或)比较运算符<(小于),>(大于),>=(大于等于),<=(小于等于),==(等于),!=(不等于)赋值运算符=,+=,-=,*=,/=,%=,&=,|=,^=,<<=,>>=其他运算符[](下标),()(函数调用),->(成员访问),,(逗号),new,delete,new[],delete[],->*(成员指针访问)177.3重载运算符的规则运算符功能.成员访问运算符.*成员指针访问运算符::域运算符sizeof长度运算符?:条件运算符表7-2C++不允许重载的运算符187.3重载运算符的规则(4)坚持4个“不能改变”。即不能改变运算符操作数的个数;不能改变运算符原有的优先级;不能改变运算符原有的结合性;不能改变运算符原有的语法结构。单目运算符重载后只能是单目运算符,双目运算符重载后依然是双目运算符。C++语言已经预先规定了每个运算符的优先级,以决定运算次序。不论怎么进行重载,各运算符之间的优先级别不会改变。C++语言已经预先规定了每个运算符的结合性。197.3重载运算符的规则(5)重载的运算符必须和用户定义的自定义类型对象一起使用,其参数至少应有一个是类对象(或类对象的引用)。也就是说,参数不能全部是C++的标准类型,以防止用户修改用于标准类型数据的运算符的性质。如果有两个参数,可以都是类对象,也可以一个是类对象,一个是C++标准类型的数据:Moneyoperator+(inta,Moneyc){return(Money(c.yuan,c.jiao,

a+c.fen);}防止用户:intoperator+(inta,intb){return(a-b);}207.3重载运算符的规则(6)重载运算符的函数不能有默认的参数,否则就改变了运算符参数的个数。(7)用于类对象的运算符一般必须重载,但有两个例外,运算符“=”和“&”可以不必用户重载。①赋值运算符(=)可以用于每一个类对象,可以利用它在同类对象之间相互赋值。②地址运算符&也不必重载,它能返回类对象在内存中的起始地址。217.3重载运算符的规则注意:(1)考虑到各种因素,双目运算符一般重载为类的友元函数,单目运算符一般重载为类的成元函数。(2)C++规定,=,(),[],->必须重载为类的成员函数,<<,>>必须重载为类的友元函数。227.4运算符重载函数作为类的成员函数和友元函数7.4.1运算符重载函数作为类的成员函数7.4.2运算符重载函数作为类的友元函数237.4.1运算符重载函数作为类的成员函数将运算符重载函数定义为类的成员函数的原型在类的内部声明格式如下:class类名{

…返回类型operator运算符(形参表);

…};在类外定义运算符重载函数的格式如下:返回类型类名::operator运算符(形参表){函数体}24【例7-2】通过运算符重载为类的成员函数来实现两个有理数对象的加、减、乘和除运算。#include<math.h>#include<iostream>usingnamespacestd;classrational//声明有理数类{public:rational(intx=0,inty=1);//构造函数

voidprint();rationaloperator+(rationala);//重载运算符"+"

rational

operator-(rationala);//重载运算符"-"private:intnum,den;voidoptimi();};//优化有理数函数25voidrational::optimi()//定义有理数优化函数{intgcd;

if(num==0)//若分子为0,则置分母为1后返回

{den=1;return;}gcd=(abs(num)>abs(den)?abs(den):abs(num));if(gcd==0)return;//若为0,则返回

for(inti=gcd;i>1;i--)//用循环找最大公约数

if((num%i==0)&&(den%i==0))break;num/=i;//i为最大公约数,将分子、分母均整除它,重新赋值

den/=i;

//若分子和分母均为负数,则结果为正,所以均改为正

if(num<0&&den<0){num=-num;den=-den;}elseif(num<0||den<0)

{//若分子和分母中只有一个为负数,则调整为分子取负,分母取正

num=-abs(num);den=abs(den);}}26voidrational::print()//输出有理数{cout<<num;

//当分子不为0且分母不为1时才显示"/分母“

if(num!=0&&den!=1)cout<<"/"<<den<<"\n";elsecout<<"\n";}7.4.1运算符重载函数作为类的成员函数27rationalrational::operator+(rationala){//“+”运算符重载函数,根据前面所列的算法写出表达式

rationalr;r.den=a.den*den;

r.num=a.num*den+a.den*num;

r.optimi();returnr;}rationalrational::operator-(rationala){//“-”运算符重载函数,根据前面所列的算法写出表达式

rationalr;r.den=a.den*den;

r.num=num*a.den-den*a.num;

r.optimi();returnr;}28intmain(){rationalr1(3,14),r2(4,14),r3,r4;r1.print();r2.print();r3=r1+r2;//使用重载了的运算符“+”r3.print();r4=r1-r2;//使用重载了的运算符“-”r4.print();return0;}7.4.1运算符重载函数作为类的成员函数297.4.2运算符重载函数作为类的友元函数将运算符重载函数定义为类的友元函数,其原型在类的内部声明格式如下:class类名{

friend返回类型operator运算符(形参表);

…};在类外定义友元运算符重载函数的格式如下:返回类型operator运算符(形参表){函数体}30【例7-3】将运算符“+”和“-”重载为适合于有理数加减法,重载函数不作为成员函数,而放在类外,作为rational类的友元函数。#include<math.h>#include<iostream.h>classrational//声明有理数类{public:

//重载函数作为友元函数friend

rational

operator+(rationala,rationalb);//重载函数作为友元函数friend

rational

operator-(rationala,rationalb);private:…};31//定义作为友元函数的重载函数rational

operator+(rationala,rationalb){rationalr;r.den=a.den*b.den;r.num=a.num*b.den+a.den*b.num;r.optimi();returnr;}//定义作为友元函数的重载函数rationaloperator-(rationala,rationalb){rationalr;r.den=a.den*b.den;r.num=a.num*b.den-a.den*b.num;r.optimi();returnr;}32intmain(){rationalr1(3,14),r2(4,14),r3,r4;r1.print();r2.print();r3=r1+r2;//使用重载了的运算符“+”

r3.print();

r4=r1-r2;

//使用重载了的运算符“-”

r4.print();return0;}7.4.2运算符重载函数作为类的友元函数337.5重载双目运算符双目运算符(或称二元运算符)有两个操作数,通常在运算符的左右两侧,如x+y,t=3,a<=b等。由于双目运算符有两个操作符,因此:

如果运算符重载函数为友元函数,则有两个参数。

如果运算符重载函数为成员函数,则可以省略一个参数。345.2向上类型转换【例7-4】定义一个Time类,用来存放做某件事所花费的时间,如3小时15分钟,分别重载运算符“+”用于求两段时间的和,重载运算符“-”用于求两段时间的差。#include<iostream.h>

classTime{

public:

Time();Time(inth,intm=0);friendTimeoperator+(Time&t1,Time&t2);friendTimeoperator-(Time&t1,Time&t2);voidShow();private:inthours,minutes;};Time::Time(){hours=minutes=0;}Time::Time(inth,intm){hours=h;minutes=m;}355.2向上类型转换voidTime::Show(){cout<<hours<<"hours,"<<minutes<<"minutes";}Timeoperator+(Time&t1,Time&t2){Timesum;sum.minutes=t1.minutes+t2.minutes;sum.hours=t1.hours+t2.hours+sum.minutes/60;sum.minutes%=60;returnsum;}Timeoperator-(Time&t1,Time&t2){Timedif;intx1,x2;

x1=t2.hours*60+t2.minutes;x2=t1.hours*60+t1.minutes;dif.minutes=(x2-x1)%60;dif.hours=(x2-x1)/60;returndif;}365.2向上类型转换intmain(){

Timet1(5,30),t2(2,48),t3,t4;cout<<"t1=";t1.Show();cout<<endl;cout<<"t2=";t2.Show();cout<<endl;cout<<"t3=t1+t2=";t3=t1+t2;t3.Show();cout<<endl;cout<<"t4=t1-t2=";t4=t1-t2;t4.Show();cout<<endl;

return0;}37指针悬挂【例3-19】默认赋值运算符重载函数引起的指针悬挂问题。【例7-5】重载赋值运算符函数解决指针悬挂问题。#include<string>#include<cassert>classString//自定义字符串类{public:

String();//默认构造函数

String(constchar*src);//带参数的构造函数~String();//析构函数constchar*ToString()const{returnstr;}

unsignedintLength()const{returnlen;}String&operator=(constString&right);private:

char*str;

unsignedintlen;};String::String()//默认构造函数{len=0;str=newchar[len+1];

str[0]='\0';}String::String(constchar*src)//带参数的构造函数{len=strlen(src);str=newchar[len+1];if(!str){cerr<<"AllocationError!\n";exit(1);}strcpy(str,src);}String::~String()//析构函数{deletestr;str=NULL;}String&String::operator=(constString&right){if(&right!=this){intlength=right.Length();if(len<length) {delete[]str;str=newchar[length+1];assert(str!=0); }for(inti=0;right.str[i]!='\0';i++)str[i]=right.str[i];str[i]='\0';len=length; } return*this;}指针悬挂intmain(){ Stringstr1("Hi!"),str2("Hello!"); cout<<"str1:"<<str1.ToString()<<endl;cout<<"str2:"<<str2.ToString()<<endl; str1=str2;cout<<"str1:"<<str1.ToString()<<endl; return0;}427.6重载单目运算符单目运算符只有一个操作数,如!a,-b,&c,*p,--i,++i等。由于单目运算符只有一个操作符,因此:

如果运算符重载函数为友元函数,则只能有一个参数。

如果运算符重载函数为成员函数,则可以省略此参数。435.2向上类型转换【例7-6】设计一个Point类,有私有数据成员x和y表示屏幕上的一个点的水平和垂直两个方向的坐标值,分别实现对自增“++”和自减“--”运算符的重载。#include<iostream>usingnamespacestd;

classPoint{public:Point();

Point(intvx,intvy);

Pointoperator++();//前置自增Pointoperator--();//前置自减

voiddisplay();private:

intx,y;};445.2向上类型转换Point::Point(){

x=0;

y=0;}Point::Point(intvx,intvy){

x=vx;

y=vy;}voidPoint::display(){

cout<<"("<<x<<","<<y<<")"<<endl;

}PointPoint::operator++();//前置自增{if(x<640)x++;//不超过屏幕的横界if(y<480)y++;//不超过屏幕的竖界

return*this;}

PointPoint::operator--();//前置自减{

if(x>0)x--;if(y>0)y--;

return*this;}455.2向上类型转换intmain(){Pointp1(10,10),p2(150,150);

cout<<"p1=";p1.display();

++p1;//测试前置自增

cout<<"++p1=";

p1.display();

cout<<"p2=";p2.display();

--p2;//测试前置自减

cout<<"-p2=";

p2.display();return0;}467.6重载单目运算符在C++中,前置单目运算符和后置单目运算符重载的主要区别就在于重载函数的形参。语法规定:(1)前置单目运算符重载为类的成员函数时没有形参,而后置单目运算符重载为类的成员函数时需要有一个int型形参。这个int型的参数在函数体内并不使用,纯粹是用来区别前置与后置,因此参数表中可以只给出类型名,没有参数名。(2)前置单目运算符重载为类的友元函数时有一个形参,即为类的对象,而后置单目运算符重载为类的友元函数时需要有两个参数,一个是类的对象,一个是int型形参。47【例7-7】在【例7-6】的基础上,增加后置单目运算符“++”和“--”的重载,其中将前置和后置的“++”运算均重载为类的成员函数,将前置和后置的“--”运算均重载为类的友元函数。7.6重载单目运算符48#include<iostream>usingnamespacestd;

classPoint{public:Point();

Point(intvx,intvy);

Pointoperator++();//重载前置自增为类的成员函数Pointoperator++(int);//重载后置自增为类的成员函数

//重载前置自减为类的友元函数friendPointoperator--(Point&p1);

//重载后置自减为类的友元函数friendPointoperator--(Point&p1,int);

voiddisplay();private:intx,y;};49Point::Point(){

x=0;

y=0;}Point::Point(intvx,intvy){

x=vx;

y=vy;}voidPoint::display(){cout<<"("<<x<<","<<y<<")"<<endl;

}PointPoint::operator++()//前置自增{if(x<640)x++;//不超过屏幕的横界if(y<480)y++;//不超过屏幕的竖界

return*this;}PointPoint::operator++(int)//后置自增{//先将当前对象通过复制构造函数临时保存起来

Pointtemp(*this);if(x<640)x++;//不超过屏幕的横界if(y<480)y++;//不超过屏幕的竖界

returntemp;}505.2向上类型转换

Pointoperator--(Point&p1)//前置自减{

if(p1.x>0)p1.x--;if(p1.y>0)p1.y--;

returnp1;}Pointoperator--(Point&p1,int)//后置自减{//先将当前对象通过复制构造函数临时保存起来Pointtemp(p1);if(p1.x>0)p1.x--;if(p1.y>0)p1.y--;returntemp;}515.2向上类型转换intmain(){Pointp1(10,10),p2(150,150),p3(20,20),p4(160,160),p5;cout<<"p1=";p1.display();++p1;//测试前置自增

cout<<"++p1=";

p1.display();cout<<"p3=";p3.display();p5=p3++;//测试后置自增

cout<<"p3++=";

p3.display();cout<<"p5=p3++=";p5.display();

cout<<"p2=";p2.display();--p2;//测试前置自减

cout<<"--p2=";p2.display();cout<<"p4=";p4.display();

p5=p4--;//测试后置自增

cout<<"p4--=";

p4.display();cout<<"p5=p4--=";

p5.display();return0;}527.7重载流插入运算符和流提取运算符在类库提供的头文件中已经对“<<”和“>>”进行了重载,使之作为流插入运算符和流提取运算符,能用来输出和输入C++标准类型的数据。用户自己定义的类型的数据,是不能直接用“<<”和“>>”来输出和输入的。如果想用它们输出和输入自己定义的类型的数据,就必须对它们进行重载。537.7重载流插入运算符和流提取运算符对“<<”和“>>”重载的函数形式如下:ostream&

operateor<<(ostream&,自定义类);istream&

operateor>>

(istream&,自定义类);在重载时要注意下面两点:(1)要对“<<”和“>>”运算符进行重载,必须重载为类的友元函数。(2)重载的友元函数的返回类型应是ostream对象或istream对象的引用,即ostream&或istream&。7.7重载流插入运算符和流提取运算符(1)要对“<<”和“>>”运算符进行重载,必须重载为类的友元函数。cout<<t;或operator<<(cout,t);voidoperator<<(ostream&out,Timer&t){

out<<t.hours<<"hours,"<<t.minutes<<"minutes";}7.7重载流插入运算符和流提取运算符(2)重载的友元函数的返回类型应是ostream对象或istream对象的引用,即ostream&或istream&。看下面的语句:intx=5,y=6;cout<<x<<y;C++从左到右读取输出语句,这意味着它等同于:(cout<<x)<<y;565.2向上类型转换【例7-8】对于Time类,在【例7-4】的基础上,增加重载流插入运算符“<<”和流提取运算符“>>”,用“cout<<”输出Time类的对象,用“cin>>”输入Time类的对象。#include<iostream.h>

classTime{public:

Time();Time(inth,intm=0);friendTimeoperator+(Time&t1,Time&t2);friendTimeoperator-(Time&t1,Time&t2);friendostream&operator<<(ostream&out,Time&t);friendistream&operator>>(istream&in,Time&t);private:inthours;intminutes;};575.2向上类型转换Time::Time(){hours=minutes=0;}Time::Time(inth,intm){hours=h;minutes=m;}Timeoperator+(Time&t1,Time&t2){Timesum;sum.minutes=t1.minutes+t2.minutes;sum.hours=t1.hours+t2.hours+sum.minutes/60;sum.minutes%=60;returnsum;}Timeoperator-(Time&t1,Time&t2){Timedif;intx1,x2;

x1=t2.hours*60+t2.minutes;x2=t1.hours*60+t1.minutes;dif.minutes=(x2-x1)%60;dif.hours=(x2-x1)/60;

returndif;}585.2向上类型转换ostream&operator<<(ostream&out,Time&t){

out<<t.hours<<"hours,"<<t.minutes<<"minutes";returnout;}istream&operator>>(istream&in,Time&t){

cout<<"inputhoursandminutes:";

in>>t.hours>>t.minutes;

returnin;}595.2向上类型转换intmain(){Timet1,t2,t3,t4;cin>>t1>>t2;cout<<"t1="<<t1<<"\n";cout<<"t2="<<t2<<"\n";t3=t1+t2;cout<<"t3=t1+t2="<<t3<<"\n";t4=t1-t2;cout<<"t4=t1-t2="<<t4<<"\n";return0;}607.8不同类型数据间的转换7.8.1系统预定义类型间的转换7.8.2转换构造函数7.8.3类型转换函数617.8.1系统预定义类型间的转换C++提供了两种转换方式:一种是隐式类型转换(或称标准类型转换)另一种是显式类型转换(或称强制类型转换)627.8.1系统预定义类型间的转换1.隐式类型转换:主要注意以下几点:(1)在C++中,将一个标准类型变量的值赋给另一个标准类型变量时,如果这两种类型兼容,则C++自动将这个值转换为接收变量的类型。(2)如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即较低类型转换为较高类型,然后再参加运算。637.8.1系统预定义类型间的转换1.隐式类型转换:主要注意以下几点:(3)当较低类型的数据转换为较高类型时,一般只是形式上有所改变,而不影响数据的实质内容,而较高类型的数据转换为较低类型时则可能有些数据丢失。(4)如果两个float型数参加运算,虽然它们类型相同,但仍要先转成double型再进行运算,结果亦为double型。647.8.1系统预定义类型间的转换2.显式类型转换C++提供显式类型转换,程序员在程序中将一种类型的数据转换为另一种指定类型的数据,其形式为:

类型名(表达式)如:

int(89.5)

其作用是将89.5转换为整型数89。65对于用户自己声明的类型,编译系统并不知道怎样进行转换。解决这个问题的关键是让编译系统知道怎样去进行这些转换,需要定义专门的函数来处理。7.8不同类型数据间的转换667.8.2转换构造函数通常把只有一个形参的构造函数,称为转换构造函数。利用它,可以将一个其他类型的数据转换成一个指定的类的对象。677.8.2转换构造函数先回顾一下以前学习过的几种构造函数:默认构造函数:Rational();//没有参数用于初始化的构造函数:Rational(intx,inty);用于复制对象的复制构造函数:Rational(Rational&c);687.8.2转换构造函数转换构造函数只有一个形参,如

Rational(intx){nume=x;deno=1;}其作用是将int型的参数x转换成Rational类的对象,将x作为有理数的分子,分母为1。697.8.2转换构造函数转换构造函数只有一个形参,如Rational(intx){nume=x;deno=1;}在类体中,可以有转换构造函数,也可以没有转换构造函数,视需要而定。以上几种构造函数可以同时出现在同一个类中,它们是构造函数的重载。编译系统会根据建立对象时给出的实参的个数与类型选择形参与之匹配的构造函数。707.8.2转换构造函数使用转换构造函数将一个指定的数据转换为类对象的方法如下:

(1)

先声明一个类。(2)在这个类中定义一个只有一个参数的构造函数,参数的类型是需要转换的类型,在函数体中指定转换的方法。(3)在该类的作用域内可以用以下形式进行类型转换:

类名(指定类型的数据)就可以将指定类型的数据转换为此类的对象。71【例7-9】基于有理数类的包含转换构造函数和运算符重载的程序。7.8.2转换构造函数72#include<iostream.h>#include<math.h>classrational//声明有理数类{public:rational();//无参构造函数rational(intx,inty);//有两个形参的构造函数rational(intx);//转换构造函数voidprint();//重载函数作为友元函数friendrationaloperator+(rationala,rationalb);private:intnum,den;voidoptimi();//有理数优化函数};73//定义无参构造函数rational::rational(){num=0;den=1;}//定义有两个形参的构造函数rational::rational(intx,inty){num=x;den=y;}//定义转换构造函数rational::rational(intx){num=x;den=1;}7.8.2转换构造函数74voidrational::print()//输出有理数{cout<<num;

//当分子不为0且分母不为1时才显示"/分母if(num!=0&&den!=1)cout<<"/"<<den<<"\n";elsecout<<"\n";}//定义作为友元函数的重载函数rationaloperator+(rationala,rationalb){rationalr;r.den=a.den*b.den;r.num=a.num*b.den+a.den*b.num;r.optimi();returnr;}voidrational::optimi()//定义有理数优化函数{…}//该函数的函数体与【例7-2】相同,在此不再列出7.8.2转换构造函数75intmain(){rationalr1(3,5),r2,r3;intn=3;r2=r1+n;r2.print();r3=n+r1;r3.print();return0;}7.8.2转换构造函数767.8.2转换构造函数说明:为了用构造函数完成类型转换,类内至少定义一个只带一个参数(或其他参数都带有默认值)的构造函数。当需要类型转换时,系统自动调用该构造函数,创建该类的一个临时对象,该对象由被转换的值初始化,从而实现了类型转换。777.8.2转换构造函数说明:转换构造函数不仅可以将一个标准类型数据转换成类对象,也可以将另一个类的对象转换成转换构造函数所在的类的对象。若有教师类Teacher,学生类Student,可以定义下面的转换构造函数将Student类转换为Teacher类。Teacher(Student&s){num=s.num;strcpy(name,);sex=s.sex;}787.8.3类型转换函数通过构造函数可以进行类型转换,但是它的转换功能受到限制。由于无法为基本类型定义构造函数,因此,不能利用构造函数把自定义类型的数据转换成基本类型的数据,它只能从基本类型(如int,float等)向自定义的类型转换。类型转换函数则可以用来把源类类型转换成另一种目的类型。797.8.3类型转换函数类型转换函数的作用是将一个类的对象转换成另一类型的数据。如果已声明了一个Rational类,可以在Rational类中这样定义类型转换函数:

operatorint(){returnnume;}807.8.3类型转换函数在类中,类型转换函数定义的一般格式为:class类名{…operator<目的类型名>(){ <函数体>}};817.8.3类型转换函数关于类型转换函数,有以下几点注意事项:(1)类型转换函数只能定义为一个类的成员函数而不能定义为类的友元函数或普通函数,因为转换的主体是本类的对象。(2)类型转换函数与普通的成员函数一样,也可以在类定义体中声明函数原型,而将函数体定义在类外部。827.8.3类型转换函数关于类型转换函数,有以下几点注意事项:(3)类型转换函数既没有参数,也不显式给出返回值类型。(4)类型转换函数中必须有

“return目的类型的数据;

”的语句,即必须送回目的类型数据作为函数的返回值。operatorint(){returnnume;}837.8.3类型转换函数关于类型转换函数,有以下几点注意事项:(5)一个类可以定义多个类型转换函数。C++编译器将根据操作数的类型自动地选择一个合适的类型转换函数与之匹配。在可能出现二义性的情况下,应显式地使用类型转换函数进行类型转换。847.8.3类型转换函数关于类型转换函数,有以下几点注意事项:(6)通常把类型转换函数也称为类型转换运算符函数,由于它也是重载函数,因此也称为类型转换运算符重载函数(或称强制类型转换运算符重载函数)。857.8.3类型转换函数下面我们为【例7-9】的有理数类再加上类型转换函数,即有理数类的声明如下:classrational{

//声明有理数类public:rational();

//无参构造函数

rational(intx,inty);

//有两个形参的构造函数

rational(intx);

//转换构造函数

operatorint(){returnnum;}

//类型转换函数

voidprint();//重载函数作为友元函数friendrationaloperator+(rationala,rationalb);private:intnum,den;voidoptimi();};intmain(){rationalr1(3,5),r2,r3;intn=3;r2=r1+n;r2.print();r3=n+r1;r3.print();return0;}867.8.3类型转换函数转换构造函数和类型转换运算符函数有一个共同的功能:

当需要的时候,编译系统会自动调用这些函数,建立一个无名的临时对象(或临时变量)。87本章小结运算符重载函数的函数名必须为关键字Operator加一个合法的运算符。当用类的成员函数实现运算符的重载时,运算符重载函数的参数(当为双目运算符时)为一个或(当为单目运算符时)没有。运算符的左操作数一定是对象,因为重载的运算符是该对象的成员函数,而右操作数是该函数的参数。88本章小结单目运算符“++”和“--”存在前置与后置问题。前置“++”格式为:

返回类型类名::operator++(){……}而后置“++”格式为:

返回类型类名::operator++(int){……}后置“++”中的参数int仅用作区分。类声明和成员函数定义的分离一般将类的声明(其中包含成员函数的声明)放在指定的头文件中,用户如果想用该类,只要把有关的头文件包含进来。由于在头文件中包含了类的声明,因此在程序中就可以用该类来定义对象。由于在类体中包含了对成员函数的声明,在程序中就可以调用这些对象的公用成员函数。90类声明和成员函数定义的分离类库有两种:一种是C++编译系统提供的标准类库;一种是用户根据自己的需要做成的用户类库,提供给自己和自己授权的人使用,这称为自定义类库。在程序开发工作中,类库是很有用的,它可以减少用户自己对类和成员函数进行定义的工作量。类声明和成员函数定义的分离类声明和成员函数定义的分离类库包括两个组成部分:(1)类声明头文件;(2)已经过编译的成员函数的定义,它是目标文件。用户只需把类库装入到自己的计算机系统中(一般装到C++编译系统所在的子目录下),并在程序中用#include命令行将有关的类声明的头文件包含到程序中,就可以使用这些类和其中的成员函数,顺利地运行程序。93ThankYou!classComplex{public:Complex();//无参构造函数

Complex(double);//转换构造函数Complex(Complex&);//拷贝构造函数

Complex(double,double);//有两个形参的构造函数Complexoperator-(Complex&,Complex&);//对“-”运算符进行重载

friendComplexoperator+(Complex&,Complex&);//对“+”运算符进行重载

friendostream&operator<<(ostream&,Complex&);//对“<<”运算符进行重载

friendistream&operator>>(istream&,Complex&);//对“>>”运算符进行重载private:doublereal,imag;};作业作业设计一个point类,有数据成员x、y,类的定义如下:classpoint {public: point(floatxx=0,floatyy=0);voiddisplay();point&operator++(); pointoperator++(int); point&operator--(); pointoperator--(int); friendpointoperator+(intn,point&pt);friendpointoperator+(point&pt,intn);friendostream&operator<<(ostream&out,pointp);friendistream&operator>>(istream&in,pointp); private: floatx,y; };作业96c.real=real+c2.real;c.imag=imag+c2.imag;c.real=this->real+c2.real;c.imag=this->imag+c2.imag;c.real=c1.real+c2.real;c.imag=c1.imag+c2.imag;c3=c1.operator+(c2);当前对象的指针7.2运算符重载的方法97运算符重载函数可以是:⑴类的成员函数

但必须要求运算表达式第一个参数(即运算符左侧的操作数)是一个类对象,而且与运算符函数的类型相同。因为必须通过类的对象去调用该类的成员函数,而且只有运算符重载函数返回值与该对象同类型,运算结果才有意义。7.2运算符重载的方法98将一个复数和一个整数相加,如c1+i;成员函数定义:ComplexComplex::operator+(int&i){returnComplex(real+i,imag);}调用:c3=c1+i;不能写成:c3=i+c1;也就是说:数学上的交换律不适用如果要计算i+c1的值,必须对“+”运算符再进行重载,重载函数为:

Complexoperator+(Complex&i,Complex&c);此时只能将重载函数作为友元函数,不能作为成员函数。99声明:friendComplexoperator+(Complex&i,Complex&c);友元函数定义:Complexoperator+(Complex&i,Complex&c){returnComplex(i+c.real,c.imag);}调用:c3=i+c1;7.2运算符重载的方法100⑵类的友元函数双目运算符重载为友元函数时,在函数的形参表列中必须有两个参数。声明:friendComplexoperator+(Complex&c1,Complex&c2);友元函数定义:Complexoperator+(Complex&c1,Complex&c2){returnComplex(c1.real+c2.real,c1.imag+c2.imag);}调用:c3=c1+c2;形参的顺序任意,不要求第一个参数必须为类对象。要求运算符左侧的操作数与第一个参数对应,运算符右侧的操作数与第二个参数对应。7.2运算符重载的方法101将一个复数和一个整数相加,如i+c1;声明:friendComplex

温馨提示

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

评论

0/150

提交评论