1、浅拷贝和深拷贝ide
所谓浅拷贝,就是由默认的拷贝构造函数所实现的对数据成员逐一赋值。若类中含有指针类型的数据,这种方式只是简单的把指针的指向赋值给新成员,但并无给新成员分配内存,所以这种方式必然会致使错误。为了解决浅拷贝出现的错误,必须显示的定义一个拷贝构造函数,使之不但复制数据成员,并且为对象分配各自的内存空间,这就是所谓的深拷贝。函数
2、浅拷贝url
浅拷贝就是由默认的拷贝构造函数所实现的数据成员逐一赋值。一般默认的拷贝构造函数可以胜任这个工做,可是若类中含有指针类型的数据,这种数据成员逐一赋值的方式将会产生错误。spa
例:指针
class Student { public: Student( char *name ,int age) //构造函数 { _name = new char [10]; //分配内存 strcpy(_name, name); _age = age; } ~Student() { delete[] _name; //释放动态内存 } void display() { cout << _name << "--" << _age << endl; } private: char *_name; int _age; }; int main() { Student st1("lisi" , 20); Student st2(st1); //调用默认的拷贝构造函数建立一个新的对象 system( "pause"); return 0; }
程序运行,建立st1时调用构造函数 ,用运算符new从堆上分配一块空间,并用_name指向这块内存空间。在执行st2语句时,由于没有定义拷贝构造函数,因此调用默认的拷贝构造函数:对象
Student(Student&st)内存
{it
_name =st._name; //并无为对象st2的数据成员_name分配新的内存空间class
_age =st._age;构造函数
}
主程序结束时,对象被逐个撤销,先撤销对象st2(由于st2后建立),撤销前先调用析构函数,用delete运算符释放所分配的内存空间;撤销对象st1时,第二次调用析构函数,由于这时_name所指向的内存空间已经被释放,企图对同一块内存空间释放两此,因此这时候程序出错。
执行过程如图:
3、深拷贝
为了解决浅拷贝出现的错误,必须显示的定义一个拷贝构造函数,使之不但能复制数据成员,并且为指针分配各自的动态内存。
例:
class Student { public: Student( char *name ,int age) //构造函数 { _name = new char [10]; //分配内存 strcpy(_name, name); _age = age; } Student( Student& st ) //自定义的拷贝构造函数 { _name = new char [10]; if (_name != NULL ) { strcpy(_name, st._name); _age = st._age; } } ~Student() { delete[] _name; //释放动态内存 } void display() { cout << _name << "--" << _age << endl; } private: char *_name; int _age; }; int main() { Student st1("lisi" , 20); Student st2(st1); //调用默认的拷贝构造函数建立一个新的对象 system( "pause"); return 0; }
在执行Student st2(st1)时调用自定义的拷贝构造函数,为st2._name分配本身的动态内存。程序的执行过程如图: