重载赋值运算符(c/c++)

本文记录了使用C++实现字符串类CMyString自定义赋值运算符的过程,包括遇到的问题及解决方案。着重讨论了字符串赋值时的内存管理问题,如如何避免内存泄漏,以及在构造函数与析构函数中正确处理字符串指针。通过实例分析加深对C++内存管理的理解。

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

最近为了准备面试,开始看《剑指Offer》这本书。的确是受到了很多启发,不管是面试技巧还是算法上。在这里记录下受到的启发和意识到的错误。
这里记录下书中提到的第一道题,我自己编写的程序,如果有什么不对的地方,各位不吝赐教。
题目是:请为CMyString该类型(有成员变量char* p_data)添加赋值运算符函数。(如果想知道完整的题目,请自行百度)。我贴出自己的代码,然后就我编写中遇到的困难做一说明。

代码如下:
CMyString.h文件

 class CMyString  
{
public:
    CMyString();
    CMyString(const CMyString& str);
    virtual ~CMyString();

    CMyString& operator=(const CMyString& old);

public:
    char* p_data;

};

MyString.cpp文件

#include "MyString.h"

CMyString::CMyString()
{
    p_data = new char[10];

}


CMyString::CMyString(const CMyString& str){
    this->p_data = str.p_data;
}

CMyString::~CMyString()
{
    if(p_data != NULL){
        delete[] p_data;
        p_data = NULL;
    }else{
        cout << "~~~~~ " << endl;
    }

}

CMyString& CMyString::operator=(const CMyString& old){


    if (this == &old)
    {
        return *this;
    }

    delete []p_data;
    p_data = NULL;

    p_data = new char[strlen(old.p_data)+1];
    strcpy(p_data, old.p_data);

    return *this;

    /*if(this != &old){
        CMyString strTemp(old);

        char* tmp = strTemp.p_data;
        strTemp.p_data = p_data;
        p_data = tmp;
    }

    return *this;*/

}

main

void main(){
    CMyString mystring1, mystring2;
    //mystring1.p_data = "wang";
    //mystring2.p_data = "zhou";
    strcpy(mystring1.p_data, "wang");
    strcpy(mystring2.p_data, "zhou");

    mystring1 = mystring2;

    cout << mystring1.p_data << "   " << mystring2.p_data << endl;
}

一开始的时候程序一直运行不了,因为用的是main函数里注释掉的那两行赋值语句,后来用了strcpy函数就可以运行了。因为如果用赋值,把p_data指向字符常量,那么重载赋值操作的时候,先delete[] p_data就会有内存错误。而且,由于在构造函数里给p_data分配了空间,并没有给它赋值,在析构的时候也不会释放内存,会导致内存泄露。
而strcpy函数,把构造函数分配的空间赋值了“wang”,这样在重载赋值操作符的delete语句便不再有问题。因此可以正常运行。
c++的内存真是一门学问,借此记录下曾经犯过的错误,希望以后不会再犯。

<think>嗯,用户想了解如何在C++中正确实现赋值运算符重载,特别是针对Date类的示例。首先,我需要回忆一下C++中运算符重载的基本规则,特别是赋值运算符的特殊要求。根据引用[3]和[4],赋值运算符必须作为类的成员函数重载,不能是全局函数。这是因为如果不在类内定义,编译器会自动生成一个默认的赋值运算符,可能会导致浅拷贝的问题,尤其是当类中有动态分配的资源时。 接下来,用户需要具体的实现方法。记得赋值运算符通常返回当前对象的引用,这样可以支持链式赋值,比如a = b = c。参数应该是const引用,避免不必要的拷贝,同时保护原对象不被修改。引用[3]中的示例代码展示了Date类的赋值运算符重载,里面还检查了自赋值的情况,这是很重要的,防止在自赋值时出现资源释放的问题。例如,在Date类中,如果自己赋值给自己,直接返回*this即可,不需要进行成员的逐一赋值。 然后,用户可能需要知道如何区分前置和后置运算符,不过这个问题主要是关于赋值运算符的,但引用[1]提到了前置++和后置++的重载方法,其中后置++需要一个int参数来区分。这可能是一个相关的知识点,但当前问题不需要深入讨论。 另外,引用[5]中的代码示例展示了默认的赋值运算符重载,如果类中没有动态资源,默认的成员逐一赋值可能已经足够。但对于有指针成员的情况,必须自定义赋值运算符来深拷贝,避免浅拷贝导致的问题。不过Date类中的成员都是int类型,所以默认的赋值运算符可能已经够用,但用户可能想了解如何显式地实现它,以便理解机制。 总结步骤应该是:1. 声明为类的成员函数;2. 参数为const引用;3. 检查自赋值;4. 逐个赋值成员变量;5. 返回*this。需要确保这些步骤在示例中清晰体现。同时,要提醒用户注意深拷贝与浅拷贝的问题,即使Date类中不需要,但其他情况下可能需要。 可能用户还会混淆赋值运算符和拷贝构造函数的区别。拷贝构造函数用于创建新对象时用已有对象初始化,而赋值运算符是在对象已存在的情况下进行赋值。因此,在实现时,拷贝构造函数需要创建新对象,而赋值运算符需要释放原有资源(如果有的话)并复制新资源,但在Date类中没有资源需要释放,所以比较简单。 最后,需要确保代码示例符合用户提供的引用中的正确格式,比如返回类型是Date&,参数是const Date&,并包含自赋值检查。这样用户可以直接参考示例代码来理解如何正确实现。</think>在C++中正确实现赋值运算符重载需要遵循以下步骤,以Date类为例说明: ### 一、基本实现方法 1. **成员函数声明** 必须声明为类的成员函数,格式为:`Date& operator=(const Date& d)` 根据C++规定,赋值运算符只能通过成员函数重载[^3][^4] 2. **参数与返回值** - 参数使用`const Date&`避免拷贝构造 - 返回`Date&`支持连续赋值操作 - 示例代码片段: ```cpp Date& operator=(const Date& d) { if(this != &d) { // 防止自赋值 _year = d._year; _month = d._month; _day = d._day; } return *this; } ```[^4] ### 二、完整Date类实现示例 ```cpp class Date { public: // 构造函数 Date(int year = 1900, int month = 1, int day = 1) : _year(year), _month(month), _day(day) {} // 拷贝构造函数 Date(const Date& d) { _year = d._year; _month = d._month; _day = d._day; } // 赋值运算符重载 Date& operator=(const Date& d) { if(this != &d) { // 自赋值检查 _year = d._year; _month = d._month; _day = d._day; } return *this; // 支持连续赋值 } void Print() { std::cout << _year << "-" << _month << "-" << _day << std::endl; } private: int _year; int _month; int _day; }; ``` ### 三、关键注意事项 1. **自赋值检查** 通过`if(this != &d)`避免对象给自己赋值的资源浪费和潜在错误[^3] 2. **返回值设计** 返回当前对象的引用(`*this`)以实现链式赋值:`d1 = d2 = d3` 3. **与拷贝构造的区别** | 特性 | 拷贝构造函数 | 赋值运算符 | |--------------|--------------------|-------------------| | 调用时机 | 对象创建时 | 已存在对象赋值时 | | 返回值 | 无 | 返回对象引用 | | 参数类型 | 必须引用 | 必须引用 | ### 四、使用示例 ```cpp int main() { Date d1(2024, 7, 12); Date d2; d2 = d1; // 调用重载赋值运算符 d2.Print(); // 输出:2024-7-12 return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值