C++重载赋值运算符

C++重载赋值运算符

一、为什么要重载赋值运算符

​ 在前面的内容中讲解 拷贝构造函数调用的时机 时说明了初始化和赋值的区别:在定义的同时进行赋值叫做初始化,定义完成以后再赋值(不管在定义的时候有没有赋值)就叫做赋值。初始化只能有一次,赋值可以有多次。

​ 当以拷贝的方式初始化一个对象时,会调用拷贝构造函数;当给一个对象赋值时,会调用重载过的赋值运算符。即使没有显式的重载赋值运算符,编译器也会以默认地方式重载它。默认重载的赋值运算符功能很简单,就是将原有对象的所有成员变量一一赋值给新对象,这和默认拷贝构造函数的类似。看下面的代码:

#include <iostream>
#include<string>
using namespace std;

class People
{
public:
    People(string name = "", int* ptr =NULL);  // 普通构造函数,
    People(const People &peo);   //显示声明拷贝构造函数
    ~People();
    void Display();
    void SetAge(int age);
private:
    string m_name;
    int* mp_age;

};


People::People(string name, int* ptr)
{
    m_name = name;
    mp_age = ptr;
}

People::People(const People &peo)
{
    this->m_name = peo.m_name;
    this->mp_age = new int(*peo.mp_age);  //重新申请一块内存来存放 age,避免两个对象使用同一块内存
}

People::~People()
{
    delete mp_age;  // 不重载赋值运算符时多次释放内存会导致崩溃。
    mp_age = NULL;
}


void People::Display()
{
    cout << m_name <<" is age "<< *mp_age << endl;
}

void People::SetAge(int age)
{
    *mp_age = age;
}

int main()
{
    int* ptr = new int(10);
    string name = "Xiao Ming";
    People people1 = People(name, ptr);
    People people2; 
    people2 = people1;  //不重载赋值运算符

    people1.Display();
    people2.Display();

    people1.SetAge(15);  // 修改 people1 age
    
    people1.Display();
    people2.Display();

    return 0;
}

/*
输出:
Xiao Ming is age 10
Xiao Ming is age 10
Xiao Ming is age 15
Xiao Ming is age 15   //修改 people1 age 之后 people2 age 也被修改了,而且调用析构函数的时候回重复释放内存导致崩溃。

 */

​ 看上面的例子修改 people1 age 之后 people2 age 也被修改了,这是因为mp_age 是一个指针,里面存放的是指向存储 age 内容的地址,不重载赋值运算符时,使用默认的赋值运算符时这是把 people1的 mp_age指针里存放的地址赋值给了people2的mp_age指针导致两个指针指向了同一块内存空间,这时候默认赋值运算符的不足就满足不了实际的需求了,需要重载赋值运算符。

二、重载赋值运算符

​ 对于简单的类,默认的赋值运算符一般就够用了,我们也没有必要再显式地重载它。但是当类持有其它资源时,例如动态分配的内存、打开的文件、指向其他数据的指针、网络连接等,默认的赋值运算符就不能处理了,我们必须显式地重载它,这样才能将原有对象的所有数据都赋值给新对象。下面我们重载赋值运算符来实现默认赋值运算符不能实现的功能。

#include <iostream>
#include<string>
using namespace std;

class People
{
public:
    People(string name = "", int* ptr =NULL);  // 普通构造函数,
    People(const People &peo);   //显示声明拷贝构造函数
    ~People();
    People& operator=(const People &peo);  // 重装赋值运算符
    void Display();
    void SetAge(int age);
private:
    string m_name;
    int* mp_age;

};

People::People(string name, int* ptr)
{
    m_name = name;
    mp_age = ptr;
}

People::People(const People &peo)
{
    this->m_name = peo.m_name;
    this->mp_age = new int(*peo.mp_age);  //重新申请一块内存来存放 age,避免两个对象使用同一块内存
}

People::~People()
{
    //释放内存,防止内存泄漏
    delete mp_age;  
    mp_age = NULL;
}

// 重装赋值运算符
People& People::operator=(const People &peo)
{
    if (this != &peo) 
    {
        this->m_name = peo.m_name;
        if (NULL != this->mp_age)
        {
            *this->mp_age = *peo.mp_age;
        }
        else
        {
            this->mp_age = new int(*peo.mp_age);
        }
    }
    return *this;
}

void People::Display()
{
    cout << m_name <<" is age "<< *mp_age << endl;
}

void People::SetAge(int age)
{
    *mp_age = age;
}

int main()
{
    int* ptr = new int(10);
    string name = "Xiao Ming";
    People people1 = People(name, ptr);
    People people2; 
    people2 = people1;

    people1.Display();
    people2.Display();

    people1.SetAge(15);  // 修改 people1 age
    
    people1.Display();
    people2.Display();

    return 0;
}
/*
输出:
Xiao Ming is age 10
Xiao Ming is age 10
Xiao Ming is age 15
Xiao Ming is age 10  //修改 people1 age 之后 people2 age 没有被修改,

 */

关于上面代码的几点说明:
  1. operator=() 的返回值类型为People &,这样不但能够避免在返回数据时调用拷贝构造函数,还能够达到连续赋值的目的,这样的语句就是连续赋值:People1 = People2 = People3;

  2. if( this != &arr)`语句的作用是「判断是否是给同一个对象赋值」:如果是,那就什么也不做;如果不是,那就将原有对象的所有成员变量一一赋值给新对象,并为新对象重新分配内存。

  3. 赋值运算符重载函数除了能有对象引用这样的参数之外,也能有其它参数。但是其它参数必须给出默认值,例如People& operator=(const People &peo, int a = 100);

  4. operator=() 的形参类型为const People &,这样不但能够避免在传参时调用拷贝构造函数,还能够同时接收 const 类型和非 const 类型的实参.

posted @ 2019-02-25 23:40 ay-a 阅读( ...) 评论( ...) 编辑 收藏
  • 6
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在C++中,重载赋值运算符是为了实现对象之间的赋值操作。通过重载赋值运算符,我们可以自定义对象的赋值行为,避免默认的浅拷贝带来的内存泄漏和重析构问题。\[1\]\[2\] 重载赋值运算符的一般形式是:`A& operator=(const A& other)`,其中A是类名,other是要赋值的对象。在重载赋值运算符时,我们需要注意以下几点: 1. 首先,要判断是否是自我赋值,即判断`this`指针是否等于`&other`。如果是自我赋值,直接返回`*this`,避免释放自己的空间后无法赋值的问题。 2. 然后,需要释放当前对象已有的资源,比如动态分配的内存。可以使用`delete`或`delete\[\]`来释放资源。 3. 接下来,根据需要,可以进行深拷贝,即重新分配内存并复制数据。可以使用`new`来分配内存,并使用`strcpy`或其他方法来复制数据。 4. 最后,返回`*this`,以支持链式赋值操作。 需要注意的是,如果没有自定义赋值运算符重载,编译器会自动生成一个默认的赋值运算符重载,这个默认的赋值运算符重载同样是等位赋值,即浅拷贝。这可能会导致内存泄漏和重析构的问题。因此,为了避免这些问题,我们需要自定义赋值运算符重载。\[2\] 总之,重载赋值运算符可以让我们自定义对象之间的赋值行为,避免浅拷贝带来的问题,并确保对象的资源得到正确释放和复制。\[1\]\[2\] #### 引用[.reference_title] - *1* *2* [C++值赋值运算符重载](https://blog.csdn.net/maoliran/article/details/51526459)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [C++ 赋值运算符重载](https://blog.csdn.net/chihiro1122/article/details/130398593)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值