Bootstrap

【C++】类和对象之赋值运算符重载

运算符重载

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类型的参数,但调用函数时该参数不用传递,编译器自动传递

;