根据上篇文章构造函数的的讲解,如果没有自定义复制构造函数,则系统会创建默认的复制构造函数,但是系统创建的默认复制构造函数只会执行“浅拷贝”,即将被拷贝对象的数据成员的值—赋值给新创建的对象,若该类的的数据成员中有指针成员,则会使新的对象的指针所指向的地址与被拷贝的对象的指针所指向的地址相同,delete该指针时则会导致两次重复delete而出错。下面是示例:
#include <iostream.h>
#include <string.h>
class Person
{
public:
//构造函数
Person(char *pn)
{
cout<<"一般构造函数被调用\n";
m_pNmae = new char[strlen(pn) + 1];
//在堆中开辟一个内存块存放p所指的字符串
if(m_Pname != NULL)
{
//如果m_pNmae不是空指针,则把形参指针pN所指的字符串复制给它
strcpy(m_pNmae,pn);
}
}
// 系统创建的默认复制构造函数,只做位模式拷贝
Person(Person & p)
{
//使两个字符串指针指向同一地址位置
m_pName = p.m_pName;
}
//c++ 析构函数调用时间:
//1、对象生命周期结束,被销毁时;
//2、delete指向对象的指针时,或delete指向对象的基类类型指针,而其基类虚构函数是虚函数时;
//3、对象i是对象o的成员,o的析构函数被调用时,对象i的析构函数也被调用。
~Person( )
{
delete m_pName;
}
private :
char * m_pName;
};
void main( )
{
Person man("lujun");
Person woman(man);
// 结果导致 man 和 woman 的指针都指向了同一个地址
// 函数结束析构时
// 同一个地址被delete两次
}
// 下面自己设计复制构造函数,实现“深拷贝”,即不让指针指向同一地址,而是重新申请一块内存给新的对象的指针数据成员
Person(Person & chs);
{
// 用运算符new为新对象的指针数据成员分配空间
m_pName=new char[strlen(p.m_pName)+ 1];
if(m_pName)
{
// 复制内容
strcpy(m_pName ,chs.m_pName);
}
// 则新创建的对象的m_pName与原对象chs的m_pName不再指向同一地址了
}
}