题目:如下为类型CMyString的声明,请为该类型添加赋值运算符函数。
class CMyString
{
public:
CMyString(char* pData = nullptr);
CMyString(const CMyString& str);
~CMyString(void);
private:
char* m_pData;
};
CMyString::CMyString(const CMyString &str)
{
int length = strlen(str.m_pData);
m_pData = new char[length + 1];
strcpy(m_pData, str.m_pData);
}
析构函数:执行delete时自动调用
经典解法
CMyString& CMyString::operator =(const CMyString &str) {
if (this == &str)
return *this;
delete m_pData;
m_pData = nullptr;
m_pData = new char[strlen(str.m_pData) + 1];
strcpy(m_pData, str.m_pData);
return *this;
}
- 是否返回本类型的引用,只有返回本类型的引用才可以连续赋值
- 参数是否为引用,倘若不是则传入参数时会调用构造函数进行值得拷贝,是引用的话可以避免这种资源的消耗
- 是否先释放内存在重新申请内存赋值,否则会造成内存泄露
是否先判断要传入的参数和当前的实例(*this)为同一个实例。否则在释放内存后再进行赋值会造成空指针异常。
使用const可以获得编译器的帮助,因而尽可能的使用const。
高阶的解法
问题:假如给m_pData重新分配内存时出现错误,会导致this不再保持有效的状态。不符合异常安全原则。
CMyString& CMyString::operator =(const CMyString &str) {
if (this != &str)
{
CMyString strTemp(str);
char* pTemp = strTemp.m_pData;
strTemp.m_pData = m_pData;
m_pData = pTemp;
}
return *this;
}