先测试下不显式定义拷贝构造函数,或拷贝构造函数声明为默认拷贝构造函数:
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