【剑指offer】01.赋值运算符函数

题目:为类型添加赋值运算符函数。

#include<iostream>
#include<cstring>


using namespace std;

class CMyString
{
	public:
		CMyString(char* pData=NULL);
		CMyString(const CMyString& str);
		~CMyString(void);
		
		CMyString& operator =(const CMyString& str);
		
		void Print();
		
	private:
		char* m_pData;
};

CMyString::CMyString(char *pData)
{
	if(pData==NULL)
	{
		m_pData=new char[1];
		m_pData[0]='\0';
	}
	else
	{
		int length=strlen(pData);
		m_pData=new char[length+1];          //字符串结尾为'\0',故比实际长大1个字节 
		strcpy(m_pData,pData);
	}
}

CMyString::CMyString(const CMyString &str)
{
	int length=strlen(str.m_pData);
	m_pData=new char[length+1];
	strcpy(m_pData,str.m_pData);
}

CMyString::~CMyString()
{
	delete[] m_pData;                                     //释放定义的数组空间 
}

CMyString& CMyString::operator =(const CMyString& str)       //返回值类型声明为该类型的引用;常量引用:a.如为实例,形参到实参会调用一次复制构造函数,降低代码效率 b.不改变传入实例状态,加上const关键字 
{
	if(this==&str)                                              //判断传入的参数和当前的实例是不是同一个实例,否则在释放实例自身的内存时会导致严重问题 
	return *this;
else
{
	CMyString strTemp(str);
	
	char* pTemp=strTemp.m_pData;                   //考虑异常安全性,当分配内存失败时能确保CMyString的实例不会被修改,创建一个临时实例,再交换临时实例和原来实例。 
	strTemp.m_pData=m_pData;
	m_pData=pTemp;
}
	
	return *this;                                             //在函数结束时返回自身的引用(*this),才可允许连续赋值(str1=str2=str3)  
	
}

//============================================测试代码=========================================
void CMyString::Print() 
{
	cout<<m_pData<<endl;
}

void Test1()
{
	cout<<"Test1 begins:"<<endl;
	
	char* text="Hello world";
	
	CMyString str1(text);
	CMyString str2;
	CMyString str3(str1);                        //构造函数 CMyString(const CMyString& str);
	str2=str1;                                   //常规赋值 
	
	cout<<"The expected str2 is: "<<text<<endl;
	cout<<"The actual str2 is:";
	str2.Print();
	
	cout<<"The expected str3 is: "<<text<<endl;
	cout<<"The actual str3 is:";
	str3.Print();
}

//赋值给自己
 void Test2()
 {
 	cout<<"Test2 begins:"<<endl;
 	
 	char* text="Hello World";
 	CMyString str1(text);
 	str1=str1;
 	
 	cout<<"The expected result is "<<text<<endl;
 	cout<<"The actual result is:";
 	str1.Print();
 }
 
 //连续赋值
 void Test3()
 {
 	cout<<"Test3 begins:"<<endl;
 	
 	char* text="Hello world";
 	
 	CMyString str1(text);
 	CMyString str2,str3;
 	str3=str2=str1;
	 
	cout<<"The expected str2 is:"<<text<<endl;
	cout<<"The actual str2 is:";
	str2.Print() ;
	
	cout<<"The expected str3 is:"<<text<<endl;
	cout<<"The actual str3 is:";
	str3.Print();
	
	 }
	 
	 int main(int argc,char* argv[])
	 {
	 	Test1();
	 	Test2();
	 	Test3();
	 	
	 	return 0;
	 	
	  } 

 注意点:

1.实现连续赋值(str3=str2=str1),故赋值函数返回值为实例自身的引用(*this)。

2.自身赋值,增加判断传入参数是否为当前实例(*this)步骤。如果事先不判断就进行赋值,那么在释放实例自身的内存的时候就会导致严重的问题:一旦释放自身的内存,那么传入参数的内存也同时被释放了。

3.把传入参数类型声明为常量引用:a.如果传入参数为实例,那么从形参到实参会调用一次复制构造函数,把参数声明为引用可以避免无谓的消耗;b.为不改变实例,增加const关键字。

4.是否释放自身已有内存,如果忘记,程序将出现内存泄漏。

运行结果:

参考:何海涛《剑指offer》

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------存疑:赋值时为何不能直接使用 m_pData=str.m_pData(实验:当这样使用时给自身赋值运行不成功)

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值