运算符重载
C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其 返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。
函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型 operator操作符(参数列表)
注意:
1.不能通过连接其他符号来创建新的操作符:比如operator@。
2.重载操作符必须有一个类类型参数。(放在全局)
3.用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不 能改变其含义。
4.作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐 藏的this。
5. .* :: sizeof ?: . 注意以上5个运算符不能重载。这个经常在笔试选择题中出现。
// 全局的operator==
class Date {
public:
Date(int year = 2024, int month = 4, int day = 22)
{
_year = year;
_month = month;
_day = day;
}
//private:
int _year;
int _month;
int _day;
};
// 这里会发现运算符重载成全局的就需要成员变量是公有的,那么问题来了,封装性如何保证?
// 这里其实可以用我们后面学习的友元解决,或者干脆重载成成员函数。
bool operator==(const Date& date1, const Date& date2)
{
return date1._year == date2._year
&& date1._month == date2._month && date1._day == date2._day;
}
赋值运算符重载
1. 赋值运算符重载格式
参数类型:const T&,传递引用可以提高传参效率
返回值类型:T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值
检测是否自己给自己赋值
返回*this :要复合连续赋值的含义
2. 赋值运算符只能重载成类的成员函数不能重载成全局函数
class Date
{
public:
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
int _year;
int _month;
int _day;
};
// 赋值运算符重载成全局函数,注意重载成全局函数时没有this指针了,需要给两个参数
Date& operator=(Date& left, const Date& right)
{
if (&left != &right)
{
left._year = right._year;
left._month = right._month;
left._day = right._day;
}
return left;
}
// 编译失败:
// error C2801: “operator =”必须是非静态成员
原因:赋值运算符如果不显式实现,编译器会生成一个默认的。此时用户再在类外自己实现 一个全局的赋值运算符重载,就和编译器在类中生成的默认赋值运算符重载冲突了,故赋值 运算符重载只能是类的成员函数。
3. 用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝。注 意:内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符 重载完成赋值。
注意:如果类中未涉及到资源管理,赋值运算符是否实现都可以;一旦涉及到资源管理则必 须要实现。
区别赋值重载和拷贝构造
//赋值拷贝 //注意如果返回值是void类型,那么无法进行连续赋值拷贝,例如d1=d2=d3 Date Date::operator=(const Date&d) { _year = d._year; _month = d._month; _day = d._day; return *this; } //拷贝构造 Date::Date(const Date&date) { _year = date._year; _month = date._month; _day = date._day; }
1.首先返回值问题,赋值重载返回值为类的类型,拷贝构造没有返回值。
2.拷贝构造:一个已经存在的对象,拷贝给另一个要创建初始化的对象
赋值拷贝:一个已经存在的对象,拷贝赋值给另一个已经存在的对象
class Date { public: Date(int year=1990,int month=1,int day=1); Date(const Date&date); Date operator=(const Date& d); private: int _year; int _month; int _day; }; //构造函数 Date::Date(int year, int month, int day) { _year = year; _month = month; _day = day; } //拷贝构造 Date::Date(const Date&date) { _year = date._year; _month = date._month; _day = date._day; } //赋值拷贝 Date Date::operator=(const Date&d) { _year = d._year; _month = d._month; _day = d._day; return *this; } int main() { Date d1; Date d2 = { 2000,2,1 }; //赋值拷贝 d2 = d1; //拷贝构造 Date d3 = d2; //赋值拷贝连续赋值 d3 = d1 = d2; return 0; }
前置++和后置++重载
class Date
{
public:
Date(int year = 1900, int month = 1, int day = 1)
{
_year = year;
_month = month;
_day = day;
}
// 前置++:返回+1之后的结果
// 注意:this指向的对象函数结束后不会销毁,故以引用方式返回提高效率
Date& operator++()
{
_day += 1;
return *this;
}
// 后置++:
// 前置++和后置++都是一元运算符,为了让前置++与后置++形成能正确重载
// C++规定:后置++重载时多增加一个int类型的参数,但调用函数时该参数不用传递,编译器
//自动传递
// 注意:后置++是先使用后+1,因此需要返回+1之前的旧值,故需在实现时需要先将this保存
//一份,然后给this+1
// 而temp是临时对象,因此只能以值的方式返回,不能返回引用
Date operator++(int)
{
Date temp(*this);
_day += 1;
return temp;
}
private:
int _year;
int _month;
int _day;
};
注意: C++规定:后置++重载时多增加一个int类型的参数,但调用函数时该参数不用传递,编译器自动传递