在前一节中曾提到过,C++中运行时的多态性主要是通过虚函数来实现的,而编译时的多态性是由函数重载和运算符重载来实现的。这一系列我将主要讲解C++中有关运算符重载方面的内容。在每一个系列讲解之前,都会有它的一些基础知识需要我们去理解。而运算符重载的基础就是运算符重载函数。所以今天主要讲的是运算符重载函数。

  1.运算符重载是对已有的运算符赋予多重含义,使同一个运算符作用域不同类型的数据导致不同行为的发生。比如

1 int i;
2 int i1=10,i2=10;
3 i=i1+i2;
4 std::cout<<"i1+i2="<<i<<std::endl;
5
6 double d;
7 double d1=20,d2=20;
8 d=d1+d2;
9 std::cout<<"d1+d2="<<d<<std::endl;

在这个程序里"+"既完成两个整形数的加法运算,又完成了双精度型的加法运算。为什么同一个运算符"+"可以用于完成不同类型的数据的加法运算?这是因为C++针对预定义基本数据类型已经对"+"运算符做了适当的重载。在编译程序编译不同类型数据的加法表达式时,会自动调用相应类型的加法运算符重载函数。但是C++中所提供的预定义的基本数据类型毕竟是有限的,在解决一些实际的问题时,往往需要用户自定义数据类型。比如高中数学里所提到的复数:

 1 class Complex //复数类
2 {
3 public:
4 double real;//实数
5 double imag;//虚数
6 Complex(double real=0,double imag=0)
7 {
8 this->real=real;
9 this->imag=imag;
10 }
11 }

假如我们建立两个复数,并用"+"运算符让它们直接相加:

1 Complex com1(10,10),com2(20,20),sum;
2 sum=com1+com2;

那么会提示没有与这些操作数匹配的 "+" 运算符的错误。这是因为Complex类类型不是预定义类型,系统没用对该类型的数据进行加法运算符函数的重载。C++就为运算符重载提供了一种方法,即运算符重载函数。其函数名字规定为operator后紧跟重载运算符。比如:operator+(),operator*()等。现在我们给上述程序声明一个加法运算符的重载函数用于完成复数的加法运算:

 #include "stdafx.h"
#include <iostream> class Complex //复数类
{
public:
double real;//实数
double imag;//虚数
Complex(double real=,double imag=)
{
this->real=real;
this->imag=imag;
}
}; Complex operator+(Complex com1,Complex com2)//运算符重载函数
{
return Complex(com1.real+com2.real,com1.imag+com2.imag);
} int main()
{
Complex com1(,),com2(,),sum;
sum=com1+com2;//或sum=operator+(com1,com2) std::cout<<"sum的实数部分为"<<sum.real<<std::endl;
std::cout<<"sum的虚数部分为"<<sum.imag<<"i"<<std::endl; return0;
}

结果:

在上述示例代码中,调用运算符重载函数时,也可以以operator+(com1,com2)的形式来调用,实际上com1+com2在程序解释时也是转化成前者一样的形式。但是直接用com1+com2的形式更加符合人的书写习惯。

  2.上述示例中的运算符重载函数是不属于任何的类,是全局的函数。因为在Complex类(复数类)中的数据成员是公有的性质,所以运算符重载函数可以访问。但如果定义为私有的呢,那该怎么办。其实,在实际的运算符重载函数声明当中,要不定义其为要操作类的成员函数或类的友元函数。

  (1)运算符重载函数作为类的友元函数的形式:

  class 类名

  {

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

  }

  类外定义格式:

  返回类型 operator运算符(参数表)

  {

    函数体

  }

友元函数重载双目运算符(有两个操作数,通常在运算符的左右两则),参数表中的个数为两个。若是重载单目运算符(只有一个操作数),则参数表中只有一参数。

i.友元函数重载双目运算符(+):

 #include "stdafx.h"
#include <iostream> class Complex //复数类
{
private://私有
double real;//实数
double imag;//虚数
public:
Complex(double real=,double imag=)
{
this->real=real;
this->imag=imag;
}
friend Complex operator+(Complex com1,Complex com2);//友元函数重载双目运算符+
void showSum();
}; Complex operator+(Complex com1,Complex com2)//友元运算符重载函数
{
return Complex(com1.real+com2.real,com1.imag+com2.imag);
} void Complex::showSum()
{
std::cout<<real;
if(imag>)
std::cout<<"+";
if(imag!=)
std::cout<<imag<<"i"<<std::endl;
} int main()
{
Complex com1(,),com2(,-),sum;
sum=com1+com2;//或sum=operator+(com1,com2)
sum.showSum();//输出复数相加结果 return0;
}

结果:

ii.友元函数重载单目运算符(++):

 #include "stdafx.h"
#include <iostream> class Point//坐标类
{
private:
int x;
int y;
public:
Point(int x,int y)
{
this->x=x;
this->y=y;
}
friend voidoperator++(Point& point);//友元函数重载单目运算符++
void showPoint();
}; voidoperator++(Point& point)//友元运算符重载函数
{
++point.x;
++point.y;
} void Point::showPoint()
{
std::cout<<"("<<x<<","<<y<<")"<<std::endl;
} int main()
{
Point point(,);
++point;//或operator++(point)
point.showPoint();//输出坐标值 return0;
}

结果:

运算符重载函数可以返回任何类型,甚至是void,但通常返回类型都与它所操作的类类型一样,这样可以使运算符使用在复杂的表达式中。比如把上述双目运算符重载函数示例代码中main()主函数里的com1+com2改为com1+com2+com2,那么结果又会不一样了。像赋值运算符=、下标运算符[]、函数调用运算符()等是不能被定义为友元运算符重载函数。同一个运算符可以定义多个运算符重载函数来进行不同的操作。

  (2)运算符重载函数作为类的成员函数的形式:

  class 类名

  {

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

  }

   类外定义格式:

  返回类型 类名:: operator 运算符(形参表)

  {

    函数体;

  }

对于成员函数重载运算符而言,双目运算符的参数表中仅有一个参数,而单目则无参数。同样的是重载,为什么和友元函数在参数的个数上会有所区别的。原因在于友元函数,没有this指针。

i.成员函数重载双目运算符(+):

#include "stdafx.h"
#include <iostream> class Complex //复数类
{
private://私有
double real;//实数
double imag;//虚数
public:
Complex(double real=,double imag=)
{
this->real=real;
this->imag=imag;
}
Complex operator+(Complex com1);//成员函数重载双目运算符+
void showSum();
}; Complex Complex::operator+(Complex com1)
{
return Complex(real+com1.real,imag+com1.imag);
} void Complex::showSum()
{
std::cout<<real;
if(imag>)
std::cout<<"+";
if(imag!=)
std::cout<<imag<<"i"<<std::endl;
} int main()
{
Complex com1(,),com2(,-),sum;
sum=com1+com2;//或sum=com1.operator+(com2)
sum.showSum();//输出复数相加结果
return0;
}

对于双目运算符而言,运算符重载函数的形参中仅为一个参数,它作为运算符的右操作数(如com2对象),而当前对象作为左操作数(如:上述中的com1对象),它是通过this指针隐含传递给成员运算符重载函数的。

ii.成员函数重载单目运算符(++):

 #include "stdafx.h"
#include <iostream> class Point//坐标类
{
private:
int x;
int y;
public:
Point(int x,int y)
{
this->x=x;
this->y=y;
}
voidoperator++();//成员函数重载双目运算符++
void showPoint();
}; void Point::operator++()
{
++x;
++y;
} void Point::showPoint()
{
std::cout<<"("<<x<<","<<y<<")"<<std::endl;
} int main()
{
Point point(,);
++point;//或point.operator++()
point.showPoint();//输出坐标值 return0;
}

对于单目运算符而言,当前对象作为运算符的操作数。

  在运算符重载运用时应该注意以下几个问题:(1)C++中只能对已有的C++运算符进行重载,不允许用户自己定义新的运算符;(2)C++中绝大部分的运算符可重载,除了成员访问运算符.,成员指针访问运算符.*,作用域运算符::,长度运算符sizeof以及条件运算符?:;(3)重载后不能改变运算符的操作对象(操作数)的个数。如:"+"是实现两个操作数的运算符,重载后仍然为双目运算符;(4)重载不能改变运算符原有的优先级;(5)重载不能改变运算符原有结合的特性。比如:z=x/y*a,执行时是先做左结合的运算x/y,重载后也是如此,不会变成先做右结合y*a;(6)运算符重载不能全部是C++中预定义的基本数据,这样做的目的是为了防止用户修改用于基本类型数据的运算符性质;(7)从上述的示例中可以看到双目运算符可以被重载为友元函数也可以重载为成员函数,但有一种情况,只能使用友元函数,是什么情况呢?我举个例子: 

 1 class Complex //复数类
2 {
3 private://私有
4 double real;//实数
5 double imag;//虚数
6 public:
7 Complex(double real=0,double imag=0)
8 {
9 this->real=real;
10 this->imag=imag;
11 }
12 Complex operator+(int x);
13 };
14
15 Complex Complex::operator+(int x)
16 {
17 return Complex(real+x,imag);
18 }
19
20 int main()
21 {
22 Complex com1(5,10),total;
23 total=com1+5;
24
25 return0;
26 }

如果我们把上述main()主函数实现部分里的total=com1+5改为total=5+com1;那么程序就会报错(没有与这些操作数匹配的 "+" 运算符),因为左操作数5不是该复数类的对象,不能调用相应的成员函数Complex operator+(int x),所以编译错误。但如果我们定义一下两个友元函数就能解决上述的问题:

  friend Complex operator+(Complex com1,int x);

  friend Complex operator+(int x,Complex com1);

  3.最后还是一样,我将用一个示例来总结一下今天所讲的内容(开发工具:vs2010): 

 #include "stdafx.h"
#include <iostream> class Complex //复数类
{
private://私有
double real;//实数
double imag;//虚数
public:
Complex(double real=,double imag=)
{
this->real=real;
this->imag=imag;
}
Complex operator+(Complex com1);//成员函数重载双目运算符+
//或friend Complex operator+(Complex com1,Complex com2);//友元函数重载双目运算符+
friend Complex operator+(Complex com1,int x);//友元函数重载双目运算符+
//或Complex operator+(int x);
friend Complex operator+(int x,Complex com1);//友元函数重载双目运算符+
void showSum();
}; Complex Complex::operator+(Complex com1)
{
return Complex(real+com1.real,imag+com1.imag);
} Complex operator+(Complex com1,int x)//左操作数类型为复数,右操作数的类型为整数
{
return Complex(com1.real+x,com1.imag);
} Complex operator+(int x,Complex com1)//左操作数类型为整数,右操作数的类型为复数
{
return Complex(x+com1.real,com1.imag);
} void Complex::showSum()
{
std::cout<<real;
if(imag>)
std::cout<<"+";
if(imag!=)
std::cout<<imag<<"i"<<std::endl;
} class Point//坐标类
{
private:
int x;
int y;
public:
Point(int x,int y)
{
this->x=x;
this->y=y;
}
friend voidoperator++(Point& point);//友元函数重载单目运算符++
Point operator++();//成员函数重载双目运算符++
void showPoint();
}; voidoperator++(Point& point)//友元运算符重载函数
{
++point.x;
++point.y;
} Point Point::operator++()
{
++x;
++y;
return*this;//返回当前对象
} void Point::showPoint()
{
std::cout<<"("<<x<<","<<y<<")"<<std::endl;
} int main()
{
//两个复数相加
std::cout<<"两个复数相加:"<<std::endl; Complex com1(,),com2(,-),sum;
sum=com1+com2;//或sum=com1.operator+(com2)
std::cout<<"(10+10i)+(20-20i)=";
sum.showSum();//输出复数相加结果 //三个复数相加
std::cout<<"三个复数相加:"<<std::endl; sum=com1+com2+com2;
std::cout<<"(10+10i)+(20-20i)+(20-20i)=";
sum.showSum(); //整数和复数相加
std::cout<<"整数和复数相加:"<<std::endl; Complex com3(,),total;
total=com3+;//或total=operator+(com1,5);
std::cout<<"(5+10i)+5=";
total.showSum(); total=+com3;//或total=operator+(5,com1);
//只能用友元函数来重载运算符
std::cout<<"5+(5+10i)=";
total.showSum(); //单目运算符++重载
std::cout<<"单目运算符++重载:"<<std::endl; //注意:下述实现部分不能只用一个++point会造成二义性
Point point(,);
//调用友元函数
operator++(point);//或++point
std::cout<<"调用友元函数:++(10,10)=";
point.showPoint();//输出坐标值 //调用成员函数
point=point.operator++();//或++point;
std::cout<<"调用成员函数:++(10,10)=";
point.showPoint(); return0;
}

结果:

出处:http://www.cnblogs.com/CaiNiaoZJ/archive/2011/08/12/2136598.html

[转]C++之运算符重载(1)的更多相关文章

  1. C++ 运算符重载时,将运算符两边对象交换问题.

    在C++进行运算符重载时, 一般来讲,运算符两边的对象的顺序是不能交换的. 比如下面的例子: #include <iostream> using namespace std; class ...

  2. C#高级编程笔记2016年10月12日 运算符重载

    1.运算符重载:运算符重重载的关键是在对象上不能总是只调用方法或属性,有时还需要做一些其他工作,例如,对数值进行相加.相乘或逻辑操作等.例如,语句if(a==b).对于类,这个语句在默认状态下会比较引 ...

  3. C++运算符重载

    C++运算符重载 基本知识 重载的运算符是具有特殊名字的函数,他们的名字由关键字operator和其后要定义的运算符号共同组成. 运算符可以重载为成员函数和非成员函数.当一个重载的运算符是成员函数时, ...

  4. 标准C++之运算符重载和虚表指针

    1 -> *运算符重载 //autoptr.cpp     #include<iostream> #include<string> using namespace std ...

  5. python运算符重载

    python运算符重载就是在解释器使用对象内置操作前,拦截该操作,使用自己写的重载方法. 重载方法:__init__为构造函数,__sub__为减法表达式 class Number: def __in ...

  6. PoEduo - C++阶段班【Po学校】-Lesson03-5_运算符重载- 第7天

    PoEduo - Lesson03-5_运算符重载- 第7天 复习前面的知识点 空类会自动生成哪些默认函数 6个默认函数    1  构造  2  析构   3  赋值  4 拷贝构造  5 oper ...

  7. 不可或缺 Windows Native (24) - C++: 运算符重载, 自定义类型转换

    [源码下载] 不可或缺 Windows Native (24) - C++: 运算符重载, 自定义类型转换 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 运算符重载 自 ...

  8. 我的c++学习(8)运算符重载和友元

    运算符的重载,实际是一种特殊的函数重载,必须定义一个函数,并告诉C++编译器,当遇到该运算符时就调用此函数来行使运算符功能.这个函数叫做运算符重载函数(常为类的成员函数). 方法与解释 ◆ 1.定义运 ...

  9. c/c++面试题(6)运算符重载详解

    1.操作符函数: 在特定条件下,编译器有能力把一个由操作数和操作符共同组成的表达式,解释为对 一个全局或成员函数的调用,该全局或成员函数被称为操作符函数.该全局或成员函数 被称为操作符函数.通过定义操 ...

  10. 实验12:Problem H: 整型数组运算符重载

    Home Web Board ProblemSet Standing Status Statistics   Problem H: 整型数组运算符重载 Problem H: 整型数组运算符重载 Tim ...

随机推荐

  1. 基于Bootstrap的Asp.net Mvc 分页的实现

    最近写了一个mvc 的 分页,样式是基于 bootstrap 的 ,提供查询条件,不过可以自己写样式根据个人的喜好,以此分享一下.首先新建一个Mvc 项目,既然是分页就需要一些数据,我这边是模拟了一些 ...

  2. 【POJ】2043.Area of Polygons

    原题戳这里 开始一小段时间的POJ计算几何练习计划(估计很快就会被恶心回去) 题解 用一条平行于y轴的扫描线,计算两条扫描线之间多少格子被覆盖了 精度可tm变态了,可能是因为题目要求的关系吧,需要上取 ...

  3. JavaScript对象参考手册

    1.array 属性: constructor 返回原型函数: length 数组个数: prototype 向对象添加属性和方法 方法: concat() 连接两个或多个数组,并返回结果: fill ...

  4. odoo发送信息到微信公众平台、企业微信

    目录 odoo发送信息到微信 @(odoo client.message.send_text) odoo发送信息到微信 在odoo平台中进行项目开发的时候有时会用到跟其他平台对接发送信息. 这里我写一 ...

  5. web服务端安全之权限漏洞

    一.权限漏洞 访问控制是指用户对系统所有访问的权限控制,通常包含水平权限,和垂直权限. 水平越权:同一角色级别的用户之间所产生的问题,如A用户可以未授权访问B用户的数据: 垂直越权:不同角色级别的用户 ...

  6. Kolla O版本部署

    Kolla O版部署和之前的版本还是有些区别的,环境还是all-in-one 基本准备: 关闭Selina和firewalld [root@kolla ~]# cat /etc/redhat-rele ...

  7. 解决mongo 端口占用问题

    在打开mongod之后如果不用了就按ctrl + c ,就不会出现以下的问题了 执行mongod报错 mongod 2016-08-03T14:31:15.691+0800 I CONTROL [in ...

  8. Ubuntu18.04 创建桌面快捷方式

    一.基本概念 Linux 系统中的Desktop Entry 文件以desktop为后缀名.Desktop Entry 文件是 Linux 桌面系统中用于描述程序启动配置信息的文件.  进入/usr/ ...

  9. ngx_lua应用最佳实践

    引子: 以下文字,是UPYUN系统开发工程师timebug在SegmentFault D-Day南京站技术沙龙上所做分享的内容要义提炼,主题为UPYUN系统开发团队在进行业务逻辑由C模块到ngx_lu ...

  10. WinlogonHack获取系统密码

    实验环境: win03 sp1 Gina.dll与Msgina.dll Gina.dll在NT/2000中交互式的登陆支持是由WinLogon调用Gina.dll实现的,Gina.dll提供了一个交互 ...