C++对象的拷贝与赋值操作

原创 2006年06月15日 13:19:00
我发现一些同事在编写一个类时,知道什么时候需要实现拷贝构造函数和赋值操作,但不知道什么时候拷贝构造函数被调用,什么时候赋值操作被调用,甚至把二者混为一谈。
 
要弄明白这个问题,最简单的做法莫过于写个测试程序试一下。不过那样做也未必是好办法,实验的结果往往导致以偏概全的结论。不如好好想一下,弄清楚其中的原理,再去写程序去验证也不迟。
 
拷贝构造函数,顾名思义,等于拷贝 + 构造。它肩负着创建新对象的任务,同时还要负责把另外一个对象拷贝过来。比如下面的情况就调用拷贝构造函数:
CString str = strOther;
 
赋值操作则只含有拷贝的意思,也就是说对象必须已经存在。比如下面的情况会调用赋值操作。
str = strOther;
 
不过有的对象是隐式的,由编译器产生的代码创建,比如函数以传值的方式传递一个对象时。由于看不见相关代码,所以不太容易明白。不过我们稍微思考一下,就会想到,既然是根据一个存在的对象拷贝生成新的对象,自然是调用拷贝构造函数了。
 
两者实现时有什么差别呢?我想有人会说,没有差别。呵,如果没有差别,那么只要实现其中一个就行了,何必要两者都实现呢?不绕圈子了,它们的差别是:
 
拷贝构造函数对同一个对象来说只会调用一次,而且是在对象构造时调用。此时对象本身还没有构造,无需要去释放自己的一些资源。而赋值操作可能会调用多次,你在拷贝之前要释放自己的一些资源,否则会造成资源泄露。
 
明白了这些道理之后,我们不防写个测试程序来验证一下我们的想法:
 
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
 
classCString
{  
public:
    CString();
    CString(constchar* pszBuffer);
    ~CString();
    CString(constCString& other);
    constCString& operator=(constCString& other);
 
private:
    char* m_pszBuffer;;
}; 
 
CString::CString()
{
    printf("CString::CString/n");
    m_pszBuffer = NULL;
 
    return;
}  
       
CString::CString(constchar* pszBuffer)
{
    printf("CString::CString(const char* pszBuffer)/n");
    m_pszBuffer = pszBuffer != NULL ? strdup(pszBuffer) : NULL;
 
    return;
}
CString::~CString()
{
    printf("%s/n", __func__);
    deletem_pszBuffer;
    m_pszBuffer = NULL;
 
    return;
}
 
CString::CString(constCString& other)
{
    if(this == &other)
    {
        return;
    }
 
    printf("CString::CString(const CString& other)/n");
    m_pszBuffer = other.m_pszBuffer != NULL ? strdup(other.m_pszBuffer) : NULL;
}
 
constCString& CString::operator=(constCString& other)
{
    printf("const CString& CString::operator=(const CString& other)/n");
 
    if(this == &other)
    {
        return *this;
    }
 
    if(m_pszBuffer != NULL)
    {
        free(m_pszBuffer);
        m_pszBuffer = NULL;
    }
         m_pszBuffer = other.m_pszBuffer != NULL ? strdup(other.m_pszBuffer) : NULL;
 
    return *this;
}
 
voidtest(CStringstr)
{
    CStringstr1 = str;
 
    return;
}
 
intmain(intargc, char* argv[])
{
    CStringstr;
    CStringstr1 = "test";
    CStringstr2str1;
    str1 = str;
 
    CStringstr3 = str3;
 
    test(str);
 
    return 0;
}

相关文章推荐

[深度探索C++对象模型] 默认构造函数和拷贝构造函数的构造操作

(1)何时编译器会合成一个默认构造函数? (2)默认构造函数做什么? (3)何时会调用拷贝构造函数? (4)什么情况下会合成拷贝构造函数呢?...

C++对象模型之默认构造函数的构造操作,拷贝构造函数同

一个类,如果没有任何的用户声明的的构造函数,那么会有一个默认的构造函数被隐式地声明出来。这个被隐式声明的构造函数,究竟什么时候被合成、被编译器合成的默认构造函数究竟执行怎么样的操作,编译器如何处理用户...

C++拷贝构造函数和赋值操作

  • 2012年01月10日 13:51
  • 3KB
  • 下载

C++对象的拷贝构造函数 赋值函数 调用情况

来源: http://www.tnove.com/?p=66     最近从新做c++,关于类的拷贝构造函数和赋值操作有点疑问,于是查询了下资料,网上也搜了下,线面总结下我的理解。先应用...

关注C++细节——含有本类对象指针的类的构造函数、析构函数、拷贝构造函数、赋值运算符的例子

本例只是对含有本类对象指针的类的构造函数、析构函数、拷贝构造函数、复制运算符使用方法的一个简单示例,以加深对构造函数和拷贝控制成员的理解。 读C++ primer 5th 第13章后加上自己的理解...

C++之 构造函数 拷贝构造函数 析构函数 赋值操作

C++之 构造函数 拷贝构造函数 析构函数 赋值操作

C++拷贝构造函数和赋值操作(operator=)

赋值运算符和复制构造函数都是用已存在的B对象来创建另一个对象A。不同之处在于:赋值运算符处理两个已有对象,即赋值前B应该是存在的;复制构造函数是生成一个全新的对象,即调用复制构造函数之前A不存在。  ...

C++拷贝构造函数和赋值操作

赋值运算符和复制构造函数都是用已存在的B对象来创建另一个对象A。不同之处在于:赋值运算符处理两个已有对象,即赋值前B应该是存在的;复制构造函数是生成一个全新的对象,即调用复制构造函数之前A不存在。  ...

使容器里对象的拷贝操作轻量而正确

转自:http://dev.wo.com.cn/bbs/viewthread.jsp?tid=66580 容器容纳了对象,但不是我们给它们的那个对象。此外,当我们从容器中获取一个对象时,所得到的对象不...

由C++ STL的vector容器中存储的对象拷贝引起的对capacity属性 的理解

【起因】 在测试C++中通过基类引用做形参实现多态的时候,写过一段很挫的测试程序,如下: Base b; Derived1 d1; Derived2 d2; Derived3 d3; vector ...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C++对象的拷贝与赋值操作
举报原因:
原因补充:

(最多只允许输入30个字)