小心,自定义拷贝函数

编译器在生成拷贝函数时只是简单地将原对象的每一个Non-static数据成员拷贝到目标对象中,这就是所说的浅拷贝。这个过程简单粗暴,如果类中有动态配置的内存,对象中包含资源,问题就会随之而产生。

首先,我们需要了解一下浅拷贝与深拷贝可能会产生的问题。

1.浅拷贝

浅拷贝是成员数据之间的一一赋值,又称为Bitwise Copy。

但可能会有这样的情况:如果对象包含资源(堆资源或文件),当进行浅拷贝时,两个资源指针会指向同一资源,就造成两个对象访问同一资源的问题,如下所示:

class Student {
public:

    Student();

    Student(const char* name);

    ~Student();

private:

    char* m_pName;
    int m_nSize;
    
};

Student strSrc("xiaohong");

Student strDst(strSrc);

写出上面这样的代码后,会出现如下情况:

浅拷贝的行为类似于memcpy。于是出现了如下问题:在释放资源的时候会产生资源归属不清的情况,这将导致程序运行出错,这是一个绝对危险的Bug。

2.深拷贝

深拷贝就是用来解决这个问题的,它会把资源也赋值一次,使对象拥有不同的资源,但其资源的内容是一样的。

以堆资源为例,就是再开辟一片堆内存,把原来的内容拷贝过来,如下图所示:

strDst与strSrc中的指针m_pName不再一样,它们分别指向了不同的内存块,但是这两个指针所指向的内存块中的内容却是一样的。

自定义拷贝函数是一种良好的编程风格,它可以阻止编译器形成默认的拷贝函数。但是在拒绝编译器默认版本的同时,你也要注意将拷贝类中所有的数据成员拷贝全了。如果你忘记拷贝某一个数据成员,你的代码并不完善,但编译器为了“报复”,是不会告诉你相关情况的,它对此依旧会表示同意。假如我们在为Student自定义拷贝函数时忽略了数据成员m_nSize,代码如下所示:

Student::Student(const Student&rhs) { 

    m_pName = new char[strlen(rhs.m_pName) + 1]; 

    strcpy(m_pName, rhs.m_pName); 
} 

Student&Student::operator=(const Student&rhs) { 

    if(this==&rhs) {

        return *this; 
    }

    if(m_pName != NULL) {

        delete m_pName; 
    }

    m_pName = new char[strlen(rhs.m_pName) + 1]; 

    strcpy(m_pName, rhs.m_pName); 

    return *this; 
}

这种实现方式大部分编译器既不会报错,也不会通过警告提示你忘记拷贝某个数据成员。这种错误一般发生在两个情形之中:

  • 对原有类进行了新设计,添加了新的数据成员,但是没有对该类的拷贝函数做适时的更新。

  • 有继承发生时,忘记对基类部分的数据进行拷贝了。如下所示:

class CollegeStudent : public Student {
public:

    CollegeStudent(const CollegeStudent& rhs)
    :m_nLenOfSchool(rhs.m_nLenOfSchool){}

    CollegeStudent&operator=(const CollegeStudent&rhs) { 

        if(this == &rhs) {

            return *this; 
        }

        m_nLenOfSchool = rhs.m_nLenOfSchool; 

        return *this; 
    }

    ~CollegeStudent(){}

private:

    int m_nLenOfSchool;

}

上述的拷贝函数貌似对所有数据成员都进行了相应的拷贝,但是它却忽略了基类部分。派生类的拷贝函数必须调用对应的基类函数,所以当你打算自己为一个派生类自定义拷贝函数时,必须注意同时拷贝其基类部分。上述CollegeStudent正确的定义方式如下所示:

class CollegeStudent : public Student {
public:

    CollegeStudent(const CollegeStudent& rhs)
    :Student(rhs), m_nLenOfSchool(rhs.m_nLenOfSchool){}

    CollegeStudent&operator=(const CollegeStudent&rhs) { 

        if(this == &rhs) {

            return *this; 
        }

        Student::operator=(rhs);

        m_nLenOfSchool = rhs.m_nLenOfSchool; 

        return *this; 
    }

    ~CollegeStudent(){}

private:

    int m_nLenOfSchool;

}

3.总结

所以,如果类中具有动态配置的内存,我们需要自行实现拷贝函数。在自定义拷贝函数时,应保证拷贝一个对象的All Parts(所有数据成员、基类部分、指针成员所指向的资源)。


个人主页:

www.codeapes.cn

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,093评论 1 32
  • 最好的生活方式,就是活出自己来,把自己充分的活出来,高兴的时候高兴,放纵的时候放纵,悲伤的时候悲伤! 太多时候,我...
    新快life阅读 80评论 0 0
  • learn effortless English 2.7,2.8review effortless English...
    西杭吟风阅读 141评论 0 0
  • 【伴读记D506】是因为我们对于小生命存有太多的宽容与期待吗?还是我格外珍惜这再次陪伴孩子成长的机会?小家伙的点滴...
    元正妈妈阅读 107评论 0 1