c++赋值运算符重载及深拷贝 浅拷贝

本文探讨了在C++中对象拷贝和赋值操作的重要性,特别是对于包含堆内存的对象。默认的浅拷贝可能会导致连续释放同一堆内存的问题,因此需要重载赋值运算符实现深拷贝。文章通过实例解释了如何重载赋值运算符以避免内存异常,并介绍了赋值运算符与拷贝构造函数的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载:https://www.cnblogs.com/zsq1993/p/5791130.html   向大佬致敬

关键词:构造函数,浅拷贝,深拷贝,堆栈(stack),堆heap,赋值运算符
摘要:
    在面向对象程序设计中,对象间的相互拷贝和赋值是经常进行的操作。
    如果对象在申明的同时马上进行的初始化操作,则称之为拷贝运算。例如:
        class1 A("af"); class1 B=A;
     此时其实际调用的是B(A)这样的浅拷贝操作。

    如果对象在申明之后,在进行的赋值运算,我们称之为赋值运算。例如:
        class1 A("af"); class1 B;
        B=A;

  此时实际调用的类的缺省赋值函数B.operator=(A);
        不管是浅拷贝还是赋值运算,其都有缺省的定义。也就是说,即使我们不overload这两种operation,仍然可以运行。
那么,我们到底需不需要overload这两种operation 呢?
        答案就是:一般,我们我们需要手动编写析构函数的类,都需要overload 拷贝函数和赋值运算符。

        
 下面介绍类的赋值运算符
1.C++中对象的内存分配方式
        在C++中,对象的实例在编译的时候,就需要为其分配内存大小,因此,系统都是在stack上为其分配内存的。这一点和C#完全不同!千 万记住:在C#中,所有类都是reference type,要创建类的实体,必须通过new在heap上为其分配空间,同时返回在stack上指向其地址的reference.
        因此,在C++中,只要申明该实例,在程序编译后,就要为其分配相应的内存空间,至于实体内的各个域的值,就由其构造函数决定了。
    例如:

class A
{
public:
    A()
    {
    }
    A(int id,char *t_name)
    {
    _id=id;
    name=new char[strlen(t_name)+1];
    strcpy(name,t_name);
    }
    private:
        char *username;
        int _id;
}

int main()

### C++ 赋值运算符重载实现深拷贝 当类成员包含指针或其他动态分配的资源时,简单的按位复制可能导致多个对象共享同一份数据副本,这被称为浅拷贝。为了避免这种情况并确保每个对象都有自己独立的数据副本,则需执行深拷贝。 对于含有指向堆内存成员变量的情况,在`operator=`函数内部不仅要复制原始对象中的数值型字段,还需要手动管理新分配的空间来存储被复制的对象内容,并释放旧有的资源以防泄漏[^3]。 下面是一个具体的例子展示如何正确处理涉及字符串类型的复杂结构: ```cpp #include <iostream> #include <cstring> class String { public: char* data; // 构造函数初始化data为NULL String(): data(nullptr) {} // 参数化构造函数用于创建新的字符数组 explicit String(const char* str){ if(str){ size_t length = strlen(str); data = new char[length + 1]; strcpy(data, str); } else{ data = nullptr; } } ~String(){ delete[] data; // 清理已分配的内存 } // 防止重复删除同一个地址上的空间以及避免悬空指针问题 String& operator=(const String& rhs){ if(this == &rhs) return *this; delete [] this->data; // 先清理当前持有的资源 if(rhs.data){ size_t length = strlen(rhs.data); this->data = new char[length + 1]; strcpy(this->data,rhs.data); }else{ this->data=nullptr; } return *this; } }; ``` 在这个案例里,每当调用赋值操作(`=`),都会触发上述定义的方法来进行安全可靠的深层复制过程,从而保证两个不同实例间不会相互干扰各自所拥有的实际资料[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值