C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。
函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型 operator操作符(参数列表)
?
下面以日期类为例子
我们知道系统的基本类型可以进行,+、-等基本运算,但是用户的自定义类型,我们不能使用系统的基本类型+和-,需要对其进行函数名重载才能使用这些操作。
运算符既可以是成员函数,也可以是非成员函数。因为我们通常将用户自定义类型的数据定义为私有,在类外不能访问私有成员。所以通常,将运算符重载函数,定义为成员函数。特别注意,成员函数的形参列表,形参的个数比实际的个数少一个。因为有一个是这个类本身。
#pragma once
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <cassert>
using namespace std;
class Date
{
public:
Date(int year = 1, int month = 1, int day = 1)
{
if (month >= 13 || month <= 0
|| day >= GetMonthDay(year, month) || day <= 0)
{
assert(false);
}
_year = year;
_month = month;
_day = day;
}
Date& operator=(const Date& d);
bool operator<(const Date& d) const;
bool operator==(const Date& d) const;
bool operator<=(const Date& d) const;
bool operator>=(const Date& d) const;
bool operator>(const Date& d) const;
bool operator!=(const Date& d) const;
Date operator+(int day);
Date& operator+=(int day);
int GetMonthDay(int year, int month);
Date& operator++();
Date operator++(int);
Date& operator-=(int day);
Date operator-(int day);
Date operator--(int);
Date& operator--();
int operator-(const Date& d);
friend ostream& operator<<(ostream& out, const Date& d);
friend istream& operator>>(istream& in, Date& d);
private:
int _year;
int _month;
int _day;
};
=的运算符重载
Date& Date::operator=(const Date& d)
{
if (this != (&d))
{
_year = d._year;
_month = d._month;
_day = d._day;
}
return *this;
}
等号运算符重载的返回值为Date& 类型,使用的是赋值后对象的自引用,可以减少拷贝,提高效率。如果this==&d,说明是自己对自己赋值,不需要进行操作。
==运算符重载
bool Date::operator==(const Date& d) const
{
return _year == d._year && _month == d._month && _day == d._day;
}
==号运算符的重载判断年、月、日是否相等就可以了。
下面就是<、<=、>=、>等运算符的重载
<运算符的重载
bool Date::operator<(const Date& d) const
{
if (_year < d._year)
{
return true;
}
else if (_year == d._year && _month < d._month)
{
return true;
}
else if (_year == d._year && _month == d._month && _day < d._day)
{
return true;
}
return false;
}
<=运算符的重载
在前面,我们已经实现过<、==的运算符重载。只需要复用前面的代码就可以实现<=的运算符重载。
bool Date::operator<=(const Date& d) const
{
return (*this < d) || (*this == d);
}
>运算符的重载
>的反面就是<=,我们对<=取反就可以完成,>运算符的重载。
bool Date::operator>(const Date& d) const
{
return !(*this <= d);
}
>=运算符的重载
>=的反面是<,只需要对<取反就可以完成>=的操作了。
bool Date::operator>=(const Date& d) const
{
return !(*this < d);
}
!=运算符的重载
!=是==的反面
bool Date::operator>(const Date& d) const
{
return !(*this <= d);
}
+=、+、-=、-运算符的重载
+=运算符的重载
int Date::GetMonthDay(int year,int month)
{
int Arry[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
if (month == 2 && (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0))
{
return Arry[month] + 1;
}
else
{
return Arry[month];
}
}
Date& Date::operator+=(int day)
{
if (day < 0)
{
return (*this) -= -day;
}
else
{
_day += day;
while (_day > GetMonthDay(_year, _month))
{
_day -= GetMonthDay(_year, _month);
++_month;
if (_month == 13)
{
_month = 1;
++_year;
}
}
return *this;
}
}
如果一个类+=day,我们先将这个类的_day+day,再判断_day的天数,有没有大于这个月的天数,如果大于了,类的月份+1,月份+1后判断,月份是否超过了12,超过了的话将月份置为1,年份+1.循环往复,就可以就出+day后的日期了。
如果+=一个负数的天数,实际上是-=一个整数的天数,我们调用-=就好了。
+运算符的重载
+运算符和+=运算符类似。
比如: int a=3;?
a+3返回结果6,但是a还是3
a+=3,返回6,a变成了6
Date Date::operator+(int day)
{
Date tmp = (*this);
tmp += day;
return tmp;
}
-=运算符的重载
Date& Date::operator-=(int day)
{
if (day < 0)
{
return (*this) += -day;
}
else
{
_day -= day;
while (_day <= 0)
{
--_month;
if (_month == 0)
{
_month = 12;
--_year;
}
_day += GetMonthDay(_year, _month);
}
return *this;
}
}
-=运算符的操作和+=的类似,我们先把类的_day-=day;如果得到一个负数,说明日期的_day不足以-day,我们需要向月份借天数,将月份--,然后_day+=借的那个月的天数。月份--之后,我们需要对月份判断是否正确,如果本来月份是1,--变成0,实际上是去年的12月,需要将_month=12,还有减少年份。之后就需要循环判断,直到_day>0为止。
-运算符的重载
-和-=运算符类似
Date Date::operator-(int day)
{
Date tmp = (*this);
tmp -= day;
return tmp;
}
++、--运算符的重载
++运算符的重载
++分为前置++与后置++
前置++运算符的重载
前置++返回的是++后的结果
Date& Date::operator++()
{
*this = *this + 1;
return *this;
}
后置++的运算符重载
我们怎么区分前置和后置呢,在形参列表加入一个int就可以说明这个是后置++了。这个int便于编译器区分。
后置++返回的是++之前的结果,++的数据仍然就行了自增。
Date Date::operator++(int)
{
Date tmp = (*this);
*this =(*this)+1;
return tmp;
}
前置--运算符重载函数
Date& Date::operator--()
{
(*this) -= 1;
return *this;
}
后置--运算符重载函数
Date Date::operator--(int)
{
Date tmp = (*this);
*this -= 1;
return tmp;
}
那么我们如何计算两个日期相差多少天呢。利用日期-时期。所以需要我们构造日期-日期的运算符重载函数。
int Date::operator-(const Date& d)
{
Date max=(*this);
Date min=d;
int flag = 1;
if (*this < d)
{
max = d;
min = (*this);
flag = -1;
}
int n = 0;
while (min != max)
{
min++;
n++;
}
return flag*n;
}
两个日期相减,我们需要先找到两个中较大的日期。
小的日期自增到与大的日期相等,自增的次数就是两个日期相差的天数。加一个flag标志用来区分日期相差的正负。
<<流插入运算符重载。我们知道流插入运算符可以打印基本数据类型。用户自定义的类型,能不能使用系统自带的流插入运算符呢?是不行的,需要我们进行函数名重载,便于支持自定义类型的打印。
ostream& operator<<(ostream& out,const Date& d)
{
out << d._year << "年" << d._month << "月" << d._day << "日" << endl;
return out;
}
流插入运算符的重载函数。不能作为类的成员函数,因为成员函数的第一个参数默认是类的this指针。但是我们使用流插入运算符是cout<<d<<endl;第一个参数为cout,所以我们不能使用成员函数作为重载函数,那么我们类外不能访问私有成员。该怎么办?我们将函数声明为友元函数,友元函数可以在类外访问私有成员。
在类内加入
为什么返回类型为ostream呢,因为这样可以做到连续的流插入,输出多个结果。
流提取运算符重载函数
我们知道>>可以输入基本的数据类型。但是日期类是我们自定义的类型。我们使用>>输入就需要对>>进行运算符重载。
与流插入运算符的重载函数一样。流提取运算符重载函数,也不能使用成员函数。写在类外将这个函数作为类的友元函数。
istream& operator>>(istream& in, Date& d)
{
in >> d._year >> d._month >> d._day;
return in;
}
在类内声明:
对<<、>>进行了运算符重载。我们就可以直接使用其操作了。