程序员求职经验分享与学习资料整理平台

网站首页 > 文章精选 正文

C++知识点 35:赋值运算符重载

balukai 2025-03-11 13:10:17 文章精选 107 ℃

#define _CRT_SECURE_NO_WARNINGS
#include
using namespace std;
#include
/*
void test()
{
Person p1("张三", 18);
Person p2("李四", 20);
p1 = p2; // 预期把 p1 改为 李四 20,但是运行报错 (报错原因见下面特别说明)
}
解决:赋值运算符重载
*/

class Person
{
public:
Person() {}
Person(char* name, int age) // 有参构造函数 初始化对象
{
this->m_Name = new char[strlen(name) + 1];
strcpy(this->m_Name, name);
this->m_Age = age;
}
~Person() // 析构函数,释放堆区空间
{
if (this->m_Name != NULL)
{
delete[] this->m_Name;
this->m_Name = NULL;
}
}

----------------------------------------------------------------------------------------------

//赋值运算符重载
Person& operator=(Person & p1) // p2 = p1 赋值运算符为双目运算符,有两个参数
{
// 释放自己(p2)堆区的数据
if (this->m_Name != NULL)
{
delete[] this->m_Name;
this->m_Name = NULL;
}
// 给自己(p2)开辟新空间,将传入的p1赋进去 // 开辟了新空间,就不会出现浅拷贝重复释放的问题了
this->m_Name = new char[strlen(p1.m_Name) + 1];
strcpy(this->m_Name, p1.m_Name);
this->m_Age = p1.m_Age;
return *this;
}

----------------------------------------------------------------------------------------------

public:
char* m_Name;
int m_Age;
};

int main()
{
Person p1("张三", 18);
Person p2("李四", 20);
p2 = p1;
// 预期把 p2 改为 张三 18,但是运行报错( 语法没有错误,其实报错的原因出现在析构函数调用delete上)
/*
特别说明:运行报错的原因
在经行对象实例化过程中,会调用4个函数:默认构造、拷贝构造、析构、 operator=
但是 operator= 只是简单的值拷贝,即 浅拷贝
当对象 p1 中有堆上属性时候, p2 = p1,就使得2个指针指向了同一块内存
当析构释放内存 p1 和 p2 时,会使同一内存会被释放2次,因此报错
赋值运算符重载的意义:将浅拷贝转变为深拷贝 (原赋值运算符是浅拷贝)
*/

cout << "p1的姓名 : " << p1.m_Name << " 年龄: " << p1.m_Age << endl;
cout << "p2的姓名 : " << p2.m_Name << " 年龄: " << p2.m_Age << endl;
return EXIT_SUCCESS;
}
说明:赋值运算符重载函数的返回值类型
1、 void 也可以,但有缺陷,如果p3 = p2 = p1, 第一次运算后会出现情况: p3 = void, 显然不符合传参类型
2、因此传回本体, Person&
再说明:比如 << = 前置++ 这一类需要实现链式编程的符号重载,返回值一般都是返回引用本体!


最近发表
标签列表