操作符重载
一、什么是操作符重载
操作符重载可以分为两部分:“操作符”和“重载”。重载是一种编译时多态,重载实际上可以分为函数重载和操作符重载。运算符重载和函数重载的不同之处在于操作符重载重载的一定是操作符。
#include <iostream> using namespace std; int main() { int a = 2 , b = 3; float c = 2.1f , d = 1.2f; cout<<"a + b = "<<a+b<<endl; cout<<"c + d = "<<c+d<<endl; return 0; }
操作符“+”完成float和int两种类型的加法计算,这就是操作符重载了。这些内置类型的操作符重载已经实现过了,但是如果现在有这样一个点类point,要实现两个点的相加,结果是横纵坐标都要相加,这时候就需要自己写一个操作符重载函数了。
#include <iostream> using namespace std; class point { double x; double y; public: double get_x() { return x; } double get_y() { return y; } point(double X = 0.0 , double Y = 0.0):x(X),y(Y){}; point operator +(point p); };
//重载操作符“+” point point::operator +(point p) { double x = this->x + p.x; double y = this->y + p.y; point tmp_p(x,y); return tmp_p; }
int main() { point p1(1.2,3.1); point p2(1.1,3.2); point p3 = p1+p2; cout<<p3.get_x()<<" "<<p3.get_y()<<endl; return 0; }
二、实现操作符重载的两种方式
操作符重载的实现方式有两种,即通过“友元函数”或者“类成员函数”。
1.友元函数重载操作符的格式:
class 类名 { friend 返回类型 operator 操作符(形参表); }; //类外定义格式: 返回类型 operator操作符(参数表) { //函数体 }
2.类成员函数实现操作符重载的格式:
class 类名 { public: 返回类型 operator 操作符(形参表); }; //类外定义格式 返回类型 类名::operator 操作符(形参表) { //函数体 }
分别用两种实现方式写point类的”+“和”-“的重载,代码如下:
#include <iostream> using std::endl; using std::cout; class point { double x; double y; public: double get_x() { return x; } double get_y() { return y; } point(double X = 0.0 , double Y = 0.0):x(X),y(Y){}; friend point operator -(point p1,point p2); point operator +(point p); };
//重载操作符“-” point operator -(point p1,point p2) { double x = p1.get_x() - p2.get_x(); double y = p1.get_y() - p2.get_y(); point p3(x,y); return p3; }
//重载操作符“+” point point::operator +(point p) { double x = this->x + p.x; double y = this->y + p.y; point tmp_p(x,y); return tmp_p; }
int main() { point p1(1.2,3.2); point p2(1.1,3.1); point p3 = p1+p2; point p4 = operator-(p1,p2); cout<<p3.get_x()<<" "<<p3.get_y()<<endl; cout<<p4.get_x()<<" "<<p4.get_y()<<endl; return 0; }
利用友元函数重载二元操作符”-“时,形式参数是两个, 而利用类成员函数时,形式参数却只有一个。这是因为类成员函数中存在this指针,相当于一个参数,所以类成员实现操作符重载需要的形式参数比原来少一 个,如果利用类成员函数实现一元操作符”-“,就不需要参数了。也正是因为这个原因,友元函数实现的操作符重载是有限制的,比如:[] ,(),->和 =不能利用友元函数实现运算符的重载。
在实际开发过程中,单目运算符建议重载为成员函数,而双目运算符建议 重载为友元函数。通常下双目运算符重载为友元函数比重载为成员函数更方便,但是有时双目运算符必须重载为成员函数,例如赋值运算符=。还有如果需要修改对 象内部的状态,一般可以选择利用类成员函数进行修改。
三、运算符重载的原则
四、为什么要进行运算符重载
关于运算符重载要遵循这么多原则,那么为什么还要进行运算符重载呢?为什么不是写一个add()函数,代替operator +()呢?个人感觉C++中之所以要支持运算符的重载是为了与内置数据类型统一操作,比如:c = a + b 和 c = add(a,b),这看起来哪个更直观一点呢,显然是前者了。同时,我们希望操作自己定义的数据类型能像操作int和double这些内置数据类型一样方便。可能举这个加法的例子有点不好,现在加入重载的运算符是[],<<,^,|等呢?这时我们要用什么成员函数代替呢?代替之后又是一种什么效果呢?会一眼就看出来这个函数要干什么吗?