一、深浅拷贝
拷贝这个词对于我们来说应该不陌生,比如我们平常的复制和粘贴就是拷贝;但是如果把拷贝这个词放到C++中来说就有一些复杂了,我们先来看一下什么是浅拷贝: 下面用字符串类来模拟实现。
class Astring
{
public:
Astring(const char* str = "")
{
_str = new char[strlen(str) + 1];
strcpy(_str, str);
}
Astring(const Astring& s)
{
_str = s._str;
}
~Astring()
{
delete[] _str;
_str = nullptr;
}
private:
char* _str;
};
int main()
{
Astring aa("hello C++");
Astring bb(aa);
return 0;
}
当我们执行以上程序的话就会失败,结果如下: 分析如下图所示: 所以我们采用浅拷贝使用同一块空间是不行了,那么怎么办呢?当然是重新开一块和别人同样大小的空间,然后再把别人空间里面的内容给拷贝过来,而这样就是所谓的深拷贝了;我们还是用字符串类来模拟实现深拷贝:
class Astring
{
public:
Astring(const char* str = "")
{
_str = new char[strlen(str) + 1];
strcpy(_str, str);
}
Astring(const Astring& s)
{
_str = new char[strlen(s._str) + 1];
strcpy(_str, s._str);
}
~Astring()
{
delete[] _str;
_str = nullptr;
}
private:
char* _str;
};
int main()
{
Astring aa("hello C++");
Astring bb(aa);
return 0;
}
分析如下图所示:
二、string类的两种写法
有了上面我们知道的深浅拷贝,所以我们明白类中的拷贝构造函数和赋值重载一定要用深拷贝来实现,不过拷贝构造函数和赋值重载还是有两种写法的。
1. 传统写法
传统写法就是要自己开辟空间自己来拷贝别人的东西,什么事情都要自己干,代码如下:
namespace cjy
{
class string
{
public:
string(const char* str = "")
:_str(new char[strlen(str) + 1])
{
strcpy(_str, str);
}
string(string& s)
:_str(new char[strlen(s._str) + 1])
{
strcpy(_str, s._str);
}
string& operator=(const string& s)
{
if (this != &s)
{
char* tmp = new char[strlen(s._str) + 1];
delete[] _str;
_str = tmp;
strcpy(_str, s._str);
}
return *this;
}
~string()
{
delete[] _str;
_str = nullptr;
}
private:
char* _str;
};
}
2. 现代写法
现代写法就是复用其它的函数,自己不用干活,交给其它函数来帮你实现,代码如下:
namespace cjy
{
class string
{
public:
string(const char* str = "")
{
_str = new char[strlen(str) + 1];
strcpy(_str, str);
}
string(const string& s)
:_str(nullptr)
{
string tmp(s._str);
std::swap(_str, tmp._str);
}
string& operator=(string s)
{
std::swap(_str, s._str);
return *this;
}
private:
char* _str;
};
}
分析如下图所示:
|