C++ 拷贝构造函数

2021-12-23  本文已影响0人  我阿郑

一、拷贝构造函数

拷贝构造函数是一种特殊的构造函数;
拷贝构造函数的含义: 以一个对象为蓝本,来构造另一个对象;

class Object
{
public:
    Object( const Object& other );
};

拷贝构造函数从来不显式调用,而是由编译器隐式地调用
在以下三种情况:

(1)定义对象

Object a;
Object b(a);  // 或写成 Object b = a;

(2)动态创建对象

Object a;
Object* p = new Object(a);

(3)函数的传值调用

void Test(Object obj);

二、 注意 : 区分“构造”与“赋值”

构造:

Object a;
Object b = a;  // 或写作  Object b(a);
// 此为“构造”,在创建对象的时候给初值,拷贝构造函数被调用

赋值:

Object a(1,2);
Object b; // 调用了构造函数创建了b对象
b = a;  // 此为“赋值”,不会调用拷贝构造函数

三、注意 : 可以访问同类对象的private成员

在拷贝构造函数,可以访问参数对象的任意成员,因为它们是同类,所以访问不受限制。

Object(const Object& other)
{
    this->a = other.a;
    this->b = other.b;
}

四、默认的“拷贝构造函数”

当没有书写拷贝构造函数时,编译器会默认提供一个拷贝构造函数。
默认的拷贝动作:将每一个成员逐个拷贝。也就是说,在多数情况下,我们无需操心,编译器会帮我们完成成员的拷贝动作。

⚠️ 注意:除非必要,不要添加拷贝构造函数。
一旦你决定了要添加拷贝构造函数,请仔细检查:

(1)所有的成员变量,要依次拷贝 (所有成员变量,不要遗漏)

(2)调用父类的拷贝构造函数 (要么不负责,要么负责全部事情)

问题:拷贝构造函数非常麻烦,容易遗漏,为什么还要自己写一个拷贝构造函数呢?

当以下情况发生时,需要添加拷贝构造函数

class Text
{
public:
    // 构造函数
    Text(const char* str)
    {
        // 申请一块内存, 保存此字符串
        m_size = strlen(str) + 1;
        m_buf = new char[m_size];
        strcpy(m_buf, str);     
    }
    ~Text()
    {
        // 释放此字符串
        delete [] m_buf;
    }
private:
    int m_size;
    char* m_buf;
};


int main()
{
      // 定义第一个对象
    Text t1("helloworld");

      // 第二个对象以t1为蓝本进行拷贝
    Text t2(t1);

    return 0;
}

对象创建

对象t1.m_buf,指向一块内存
对象t2拷贝了t1, t2.m_buf指向了同一块内存

对象析构

对象t1析构, delete [] m_buf;
对象t2析构,delete [] m_buf;出错,此块内存已经被delete

⚠️ 错误的根本原因:应该拷贝其数据,而不是拷贝其指针

解决办法

(1) 正规解决方法 : 添加拷贝构造函数,拷贝其具体的数据
Text(const Text& other)
{
    m_size = other.m_size;
    m_buf = new char[m_size];
    strcpy(m_buf, other.m_buf);
}

此种情况称为“深度拷贝”

(2) 省事的办法

禁止用户进行拷贝构造,将拷贝构造函数设定为private,一般不这么做

private:
    Text(const Text& other)
    {
    }
上一篇 下一篇

猜你喜欢

热点阅读