剑指offer 面试题1:赋值运算符函数【C++版本】

题目总结与代码归档:
【剑指offer-2】题目目录【C++版本】

GitHub代码路径: GitHub

面试题1

赋值运算符函数

题目: 如下为类型CMyString的声明,请为该类型添加赋值运算符函数。

class CMyString {
public:
    //构造函数
    CMyString(char *pData = nullptr);
    //拷贝构造函数
    CMyString(const CMyString &str);
    ~CMyString(void);
private:
    char *m_pData;
};

考察点

1、是否把返回值的类型声明为该类型的引用,并在函数结束前返回实例自身的引用(*this)。返回引用才允许连续赋值
2、把形参声明成该类型的常量引用,避免不必要的拷贝构造;
3、如果有必要,需要释放实例自身已经拥有的内存;
4、判断传入的参数和当前的实例(*this)是不是同一个实例(判断自赋值的情况);

进阶
判断会不会因为内存不足而导致new失败,解决办法有二 :
5、先new分配空间,然后再delete释放。只有在分配空间成功后才释放原有内容,即分配失败时我们可以保证原有内容保存下来;
6、先使用拷贝构造函数创建一个临时实例,然后交换临时实例和自身。临时实例在拷贝赋值函数之后自动析构了,且如果临时实例分配内存失败,也不会改变原有内容。

代码实现

方法一

// 方法一:
// 赋值运算符函数  考察点 1
CMyString& CMyString::operator = (const CMyString& str) // 考察点 2 
{
	printf("这是拷贝赋值运算符!\n");

	if (this == &str)	// 
		//检查自赋值的情况	考察点 4
		return *this;

	delete[]m_pData;	//  考察点 3
	m_pData = nullptr;

	m_pData = new char[strlen(str.m_pData) + 1];
	strcpy(m_pData, str.m_pData);

	return *this;	// 考察点 1
}

方法二

// 方法二:
CMyString &CMyString::operator=(const CMyString &str) {

	if (this != &str) {
		//调用拷贝构造函数
		CMyString strTmp(str);	// 进阶 6

		char *pTmp = strTmp.m_pData;
		strTmp.m_pData = m_pData;
		m_pData = pTmp;
	}
	// if 语句结束 strTmp 生命周期结束 自动调用 strTmp的析构函数
	return *this;
}

#include <cstring>
#include <iostream>
using namespace std;

class CMyString
{
public:
	CMyString(char* pData = nullptr);
	//拷贝构造函数
	CMyString(const CMyString& str);
	~CMyString(void);

	CMyString& operator = (const CMyString& str);

	char *getData();
private:
	char* m_pData;
};

//构造函数 1   char *pData
CMyString::CMyString(char *pData)
{
	cout << "构造函数 1" << endl;
	if (pData == nullptr)	// 参数检测
	{
		m_pData = new char[1];
		m_pData[0] = '\0';
	}
	else
	{
		//  strlen 计算字符串的长度,以结束符 0x00 为字符串结束(不包含)
		//可参见 https://www.cnblogs.com/BeyondAnyTime/archive/2012/05/28/2521460.html
		int length = strlen(pData);	
		m_pData = new char[length + 1];	// new char[]
		strcpy(m_pData, pData);	//拷贝
	}
}

//构造函数 2  const CMyString &str //参数使用const的引用
CMyString::CMyString(const CMyString &str)
{
	cout << "这是构造函数 2" << endl;
	int length = strlen(str.m_pData);
	m_pData = new char[length + 1];
	strcpy(m_pData, str.m_pData);
}
// 析构函数
CMyString::~CMyString()
{
	delete[] m_pData;	// delet[]
}

char *CMyString::getData() {
	return m_pData;
}

 方法一:
 赋值运算符函数  考察点 1
//CMyString& CMyString::operator = (const CMyString& str) // 考察点 2 
//{
//	cout << "这是拷贝赋值运算符!方法一" << endl;
//
//	if (this == &str)	// 
//		//检查自赋值的情况	考察点 4
//		return *this;
//
//	delete[]m_pData;	//  考察点 3
//	m_pData = nullptr;
//
//	m_pData = new char[strlen(str.m_pData) + 1];
//	strcpy(m_pData, str.m_pData);
//
//	return *this;	// 考察点 1
//}

// 方法二:
CMyString &CMyString::operator=(const CMyString &str) {
	cout << "这是拷贝赋值运算符!方法二\n" << endl;
	if (this != &str) {
		//调用拷贝构造函数
		CMyString strTmp(str);	// 进阶 6

		char *pTmp = strTmp.m_pData;
		strTmp.m_pData = m_pData;
		m_pData = pTmp;
	}
	// if 语句结束 strTmp 生命周期结束 自动调用 strTmp的析构函数
	return *this;
}

int main()
{
	char *tmp = "hello world +++";
	CMyString myStr(tmp);	// 
	cout << "myStr: " << myStr.getData() << endl;

	CMyString otherOne = myStr;
	cout << "otherOne: " << otherOne.getData() << endl;

	char *tmp2 = "hello world ---";
	CMyString myStr2(tmp2);
	cout << "myStr2: " << myStr2.getData() << endl;

	myStr2 = otherOne;
	cout << "myStr2 after operator \"=\": " << myStr2.getData() << endl;

	char* text = "Hello world";
	CMyString str1(text);
	CMyString str2, str3;
	str3 = str2 = str1;
	cout << "str3 = " << str3.getData()<< " str2 = " << str2.getData() << " str1 = " << str1.getData() << endl;

	system("pause");
	return 0;
}

参考资料:


GitHub链接:
https://github.com/lichangke
CSDN首页:
https://me.csdn.net/leacock1991
欢迎大家来一起交流学习

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

墨1024

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值