前言
运算符默认都是给内置类型用的,自定义类型想用,得自己进行运算符重载。
运算符重载:自己写一个函数实现运算符的行为
提示:以下是本篇文章正文内容,下面案例可供参考
一、定义一个日期类用于测试
class Date
{
public:
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
void print()//输出日期
{
cout << _year << "/" << _month << "/" << _day << endl;
}
private:
int _year;
int _month;
int _day;
};
正常情况下如果想比较两个日期大小是无法实现的,这是因为运算符默认都是给内置类型用的。
int main()
{
Date d1(2022, 2, 21);
Date d2(2022, 2, 23);
Date d3(2022, 2, 24);
//d1 == d2;直接比较会导致无法编译
return 0;
}
二、重载运算符==
函数名:operator加上运算符
参数:有几个操作数就有几个参数,参数类型就是要操作对象的类型
返回值:看运算符运算后的返回值是什么
//存在this指针,要少传一个参数
bool operator==(const Date& x)//引用节省空间,const保护实参
{
return _year == x._year &&
_month == x._month &&
_day == x._day;
}
公有函数无法访问私有变量,所以运算符重载要写在类内当作成员函数
三、日期赋值=
- 参数类型
- 返回值
- 检测是否自己给自己赋值
- 返回 * this
- 一个类如果没有显式定义赋值运算符重载,编译器也会生成一个,完成对象按字节序的值拷贝。
如果不写,会默认生成赋值重载,和拷贝构造行为类似,内置类型会完成值拷贝,自定义类型成员会调用他的赋值重载
//赋值重载
Date operator=(const Date& d)
{
_year = d._year;
_month = d._month;
_day = d._day;
return *this;//支持连续赋值,this是当前对象的别名例如d2 = d3,this指向d2
}
测试结果
int main()
{
Date d1(2022, 2, 21);
Date d2(2022, 2, 23);
Date d3(2022, 2, 24);
d1 == d2;
//d1.operator== (d2);//可以调用但可读性差
//d1 == d2;//编译器自动转换为 d1.operator== (d2);
//赋值重载
d1 = d3;
d1.print();
//赋值运算符重载:用于两个已经定义出的对象间的拷贝赋值
//拷贝构造:一个对象准备定义时,用另一个对象来初始化他
Date d4(d3);
d4.print();
Date d5 = d3;//这里是拷贝构造,只要是创建时定义就是拷贝构造,注意区分赋值重载。
d1 = d3 = d2;//连续赋值
d1.print();
return 0;
}
总结
注意:
- 不能通过连接其他符号来创建新的操作符:比如operator@
- 重载操作符必须有一个类类型或者枚举类型的操作数
- 用于内置类型的操作符,其含义不能改变,例如:内置的整型 + ,不 能改变其含义
- 作为类成员的重载函数时,其形参看起来比操作数数目少1成员函数的操作符有一个默认的形参this,限定为第一个形参
- .*、::、sizeof 、 ? : 、.注意以上5个运算符不能重载。这个经常在笔试选择题中出现。
总结:
- 栈:构造,析构,拷贝构造,赋值重载都需要自己写
- 日期类除构造函数外都可以使用编译器默认