c++ 拷贝构造函数
先测试下不显式定义拷贝构造函数,或拷贝构造函数声明为默认拷贝构造函数:
void testCopyConstructor() {class MyStr {private:char* str;int len;public:MyStr(const char* s = "") {len = strlen(s);str = new char[len + 1];strcpy_s(str, len + 1, s);}MyStr(const MyStr& other) = default; // 默认拷贝构造函数,或者去掉这句代码效果一样。// 析构函数会报错,因为调用拷贝构造函数后,另一个str和这个str地址相同,是同一个字符串,重复释放内存会报错!//~MyStr() { //delete[] str; // 释放数组//str = nullptr;//}void print() { cout << str << " len: " << len << " ,address of str: " << (void*)str<< " ,address of len: " << &len<< endl; }};MyStr s1 = "我是9527";MyStr s2 = s1; // 调用拷贝构造函数. 没有显示声明拷贝构造函数,会有默认的拷贝构造函数s1.print();s2.print();
}
打印:
可见,字符串这种指针类型数据其地址没变,还是原字符串,即浅拷贝。而int类型这种类型是值拷贝,地址变了。 析构函数被我注释了,此时调用析构函数会报错,因为str这个同一个字符串被重复释放内存了。
定义拷贝构造函数,代码:
void testCopyConstructor() {class MyStr {private:char* str;int len;public:MyStr(const char* s = "") {len = strlen(s);str = new char[len + 1];strcpy_s(str, len + 1, s);}// 拷贝构造函数(深拷贝)MyStr(const MyStr& other) {len = strlen(other.str);str = new char[len + 1];strcpy_s(str, len + 1, other.str);}~MyStr() { delete[] str; // 释放数组str = nullptr;}void print() { cout << str << " len: " << len << " ,address of str: " << (void*)str<< " ,address of len: " << &len<< endl; }};MyStr s1 = "我是9527";MyStr s2 = s1; // 调用拷贝构造函数. 没有显示声明拷贝构造函数,会有默认的拷贝构造函数s1.print();s2.print();
}
打印:
可见,拷贝构造函数中,将字符串也拷贝,比较俩个对象,字符串地址不一样,说明是深拷贝。调用析构函数也不报错了。ok