一、在OC中:
淺拷貝:淺拷貝就是對內(nèi)存地址的復(fù)制犁河,讓目標(biāo)對象指針和源對象指向同一片內(nèi)存空間,當(dāng)內(nèi)存銷毀的時候棋嘲,指向這片內(nèi)存的幾個指針需要重新定義才可以使用店溢,要不然會成為野指針叁熔。
淺拷貝就是拷貝指向原來對象的指針,使原對象的引用計數(shù)+1床牧,可以理解為創(chuàng)建了一個指向原對象的新指針而已荣回,并沒有創(chuàng)建一個全新的對象。
深拷貝:深拷貝是指拷貝對象的具體內(nèi)容戈咳,而內(nèi)存地址是自主分配的心软,拷貝結(jié)束之后,兩個對象雖然存的值是相同的著蛙,但是內(nèi)存地址不一樣删铃,兩個對象也互不影響,互不干涉踏堡。
深拷貝就是拷貝出和原來僅僅是值一樣猎唁,但是內(nèi)存地址完全不一樣的新的對象,創(chuàng)建后和原對象沒有任何關(guān)系顷蟆。
總結(jié):
1:可變對象的copy和mutableCopy方法都是深拷貝(區(qū)別完全深拷貝與單層深拷貝) 诫隅。
2:不可變對象的copy方法是淺拷貝,mutableCopy方法是深拷貝帐偎。
3:copy方法返回的對象都是不可變對象逐纬。
二、在C++中:
對一個已知對象進(jìn)行拷貝削樊,編譯系統(tǒng)會自動調(diào)用一種構(gòu)造函數(shù)——拷貝構(gòu)造函數(shù)豁生,如果用戶未定義拷貝構(gòu)造函數(shù),則會調(diào)用默認(rèn)拷貝構(gòu)造函數(shù)漫贞。
//student.h
#ifndef STUDENT_H
#define STUDENT_H
class Student
{
? ? ? ? private:
? ? ? ? ? ? ? int num;
? ? ? ? ? ? ? char *name;
? ? ? ? public:
? ? ? ? ? ? ? ? Student();
? ? ? ? ? ? ? ? ~Student();
};
#endif
//student.cpp
#include "student.h"
#include <iostream>
using namespace std;
Student::Student()
{
? ? ? name = new char(20);
? ? ? cout << "Student" << endl;
}
Student::~Student()
{
? ? ? ? cout << "~Student " << (int)name << endl;
? ? ? ? delete name;
? ? ? ? name = NULL;
}
//main.cpp
#include <iostream>
#include "student.h"
int main()
{
? ? ? ? Student s1;
? ? ? ? Student s2(s1);//Student s2 = s1;//復(fù)制對象
? ? ? ? return 0;
}
執(zhí)行結(jié)果:調(diào)用一次構(gòu)造函數(shù)甸箱,調(diào)用兩次析構(gòu)函數(shù),兩個對象的指針成員所指內(nèi)存相同
name指針被分配一次內(nèi)存迅脐,但是程序結(jié)束時該內(nèi)存卻被釋放了兩次摇肌,會造成內(nèi)存泄漏問題!
這是由于編譯系統(tǒng)在我們沒有自己定義拷貝構(gòu)造函數(shù)時仪际,會在拷貝對象時調(diào)用默認(rèn)拷貝構(gòu)造函數(shù)围小,進(jìn)行的是淺拷貝!即對指針name拷貝后會出現(xiàn)兩個指針指向同一個內(nèi)存空間树碱。
所以肯适,在對含有指針成員的對象進(jìn)行拷貝時,必須要自己定義拷貝構(gòu)造函數(shù)成榜,使拷貝后的對象指針成員有自己的內(nèi)存空間框舔,即進(jìn)行深拷貝,這樣就避免了內(nèi)存泄漏發(fā)生。
?自己定義拷貝構(gòu)造函數(shù):
在student.h 申明拷貝構(gòu)造函數(shù) :
Student(constStudent &s);//拷貝構(gòu)造函數(shù)刘绣,const防止對象被改變
在student.cpp 實現(xiàn)拷貝構(gòu)造函數(shù) :
Student::Student(constStudent &s)
{
? ? ? ? name =newchar(20);
? ? ? ? memcpy(name, s.name, strlen(s.name));
? ? ? ? cout <<"copy Student "<< endl;
}
執(zhí)行結(jié)果:調(diào)用一次構(gòu)造函數(shù)樱溉,一次自定義拷貝構(gòu)造函數(shù),兩次析構(gòu)函數(shù)纬凤。兩個對象的指針成員所指內(nèi)存不同福贞。
總結(jié):淺拷貝只是對指針的拷貝,拷貝后兩個指針指向同一個內(nèi)存空間停士,深拷貝不但對指針進(jìn)行拷貝挖帘,而且對指針指向的內(nèi)容進(jìn)行拷貝,經(jīng)深拷貝后的指針是指向兩個不同地址的指針恋技。
再說幾句:
當(dāng)對象中存在指針成員時拇舀,除了在復(fù)制對象時需要考慮自定義拷貝構(gòu)造函數(shù),還應(yīng)該考慮以下兩種情形:
1.當(dāng)函數(shù)的參數(shù)為對象時蜻底,實參傳遞給形參的實際上是實參的一個拷貝對象骄崩,系統(tǒng)自動通過拷貝構(gòu)造函數(shù)實現(xiàn);
2.當(dāng)函數(shù)的返回值為一個對象時薄辅,該對象實際上是函數(shù)內(nèi)對象的一個拷貝刁赖,用于返回函數(shù)調(diào)用處。