[C++系列] 19. 拷贝构造函数

1. 概念

在现实生活中,可能存在一个与你一样的自己,我们称其为双胞胎。

那在创建对象时,可否创建一个与一个对象一某一样的新对象呢?
构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。

2. 特征

拷贝构造函数也是特殊的成员函数,其特征如下: 
1)拷贝构造函数是构造函数的一个重载形式。
2)拷贝构造函数的参数只有一个且必须使用引用传参,使用传值方式会引发无穷递归调用,拷贝函数的参数一直赋值,拷贝工作未完成。

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;
    }
private:
    int _year;
    int _month;
    int _day;
};
 
int main() {
    Date d1;
    Date d2(d1);
 
    return 0;
}

3)若未显示定义,系统生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,对象里有什么就进行拷贝,而资源不会进行拷贝,均指向同一个空间。这种拷贝我们叫做浅拷贝,或者值拷贝。浅拷贝:值拷贝对象本身的内容。深拷贝:拷贝对象本身内容+申请的资源。

class Date {
public:
    Date(int year = 1900, int month = 1, int day = 1) {
        _year = year;
        _month = month;
        _day = day;
    }
private:
    int _year;
    int _month;
    int _day;
};
 
int main() {
    Date d1;
    // 这里d2调用的默认拷贝构造完成拷贝,d2和d1的值也是一样的。
    Date d2(d1);
    return 0;
}

4)那么编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,我们还需要自己实现吗?当然像日期类这样的类是没必要的。那么下面的类呢?验证一下试试?到底自己需不需要实现拷贝构造主要查看定义的对象有没有申请资源,若定义的对象申请了资源得自己实现拷贝构造,编译器自动实现的仅为浅拷贝。

// 这里会发现下面的程序会崩溃掉?这里就需要我们以后讲的深拷贝去解决。
class String {
public:
    String(const char* str = "jack") {
        _str = (char*)malloc(strlen(str) + 1);
        strcpy(_str, str);
    }
 
    ~String() {
        cout << "~String()" << endl;
        free(_str);
    }
private:
    char* _str;
};
 
int main() {
    String s1("hello");
    String s2(s1);
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ypuyu

如果帮助到你,可以请作者喝水~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值