對深拷貝和淺拷貝的一些學習心得

weixin_34116110發表於2016-08-26

摘要 : 文章參考http://www.cocoachina.com/ios/20160803/17275.html

一開始我一直以為拷貝方式只有兩種,並且對這兩種的拷貝方式也是分不清,然後就在網上學習了一下,寫一下自己的心得體會,給大家分享一下.

首先 . OC物件的拷貝方式不是兩種而是三種.分別是

淺拷貝(shallow copy):在淺拷貝操作時, 對於被複制的物件的每一層都是指標複製,並沒有對實體地址進行復制,所以並不會重新開闢新的空間.

深複製(one-level-deep copy):在深複製操作的時候,是把物件的給複製過來,至少有一層是深複製.其實也不全是深複製,如果資料有很多層次,它就只複製了第一層,而第二層還是淺複製.

完全複製(real - deep copy):完全複製操作的時候,就是對於物件的每一層都進行了複製,不僅實體地址複製,物件也複製.這才是真正的深拷貝.

圖片來源http://www.cocoachina.com/ios/20160803/17275.html

2491655-9f8c926814d5d75b.jpg

從圖片可以看出 ,我發現可變的物件 雖然只是用了copy ,但發現它並不是淺拷貝,而是深拷貝.但引數的型別竟然發生了改變,具體原因我也不清楚 ,如有大神 知道,可以幫小弟講解一下.


2491655-ec08c626a4c302c0.jpg

理解深複製(mutableCopy)

說多無益,程式碼才是王道.

NSMutableArray *array = [NSMutableArray arrayWithObjects:

[NSMutableString stringWithString:@"a"],

[NSMutableString stringWithString:@"b"],

[NSMutableString stringWithString:@"c"],

[NSMutableString stringWithString:@"d"],

nil];

NSMutableArray *array1 = [NSMutableArray arrayWithObjects:

[NSMutableString stringWithString:@"A"],

[NSMutableString stringWithString:@"B"],

[NSMutableString stringWithString:@"C"],

[NSMutableString stringWithString:@"D"],

array, nil];

NSMutableArray *array2 ;

NSMutableString *str;

array2 = [array1 mutableCopy];

str = array1[4][1];

[str appendString:@"-----1"];

NSLog(@"array2------%@",array2);

NSLog(@"array1------%@",array1);

列印出來的結果卻讓人想不明白

2016-08-26 12:08:38.277 深淺拷貝[1117:64610] array2------(

A,

B,

C,

D,

(

a,

"b-----1",

c,

d

)

)

2016-08-26 12:08:38.278 深淺拷貝[1117:64610] array1------(

A,

B,

C,

D,

(

a,

"b-----1",

c,

d

)

)


試了一下又開了一下原文講的才知道,原來這就是深複製和完全複製的區別.

對於深複製,我一直認為,深複製就是對於原有物件的內容直接克隆過去,但程式碼顯示的結果卻讓我產生疑惑,很是不明白,原來它只是複製一層物件,而不會複製第二層甚至更深層次的物件,其實對於這句話我是有疑問的,既然是複製了第一層,那麼這第一層是說的A,B,C,D呢還是就是原本array1的層次,就是一個空的物件,但看程式碼執行的結果,我的想法應該是對的,這個層次,對於我來說,好像並不是很清楚,同樣如果有大神知道,可以講解一下.

程式碼array2 = [array1 mutableCopy]; 只是對陣列array1 本身進行內容的拷貝.但裡面的字串物件沒有進行內容的拷貝,而是進行的淺複製,只是指標的複製,物件還是公用的所以改變一個也會改變所有.

看來解決這個問題只能再看原文章了

換了複製方式的程式碼 結果還是可人的.

array2 = [[NSMutableArray alloc]initWithArray:array1 copyItems:YES];

顯示結果是

2016-08-26 12:23:25.787 深淺拷貝[1195:69165] array2------(

A,

B,

C,

D,

(

a,

b,

c,

d

)

)

2016-08-26 12:23:25.787 深淺拷貝[1195:69165] array1------(

"A-----1",

B,

C,

D,

(

a,

b,

c,

d

)

)


發現沒有問題 但我總感覺 還是有問題,

str = array1[4][1];

方法一變發現列印的結果又和上面的一樣

2016-08-26 12:24:42.143 深淺拷貝[1218:70070] array2------(

A,

B,

C,

D,

(

a,

"b-----1",

c,

d

)

)

2016-08-26 12:24:42.144 深淺拷貝[1218:70070] array1------(

A,

B,

C,

D,

(

a,

"b-----1",

c,

d

)

)


唉, 還是失敗了,array2 = [[NSMutableArray alloc]initWithArray:array1 copyItems:YES];僅僅只是複製了一層,而下面的一層卻沒有複製.

看來我的試一下完全複製這個方法了.原來完全複製是歸檔和解檔啊.

array2 = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:array1]];

輸出的結果是

2016-08-26 12:30:12.292 深淺拷貝[1265:72371] array2------(

A,

B,

C,

D,

(

a,

b,

c,

d

)

)

2016-08-26 12:30:12.293 深淺拷貝[1265:72371] array1------(

A,

B,

C,

D,

(

a,

"b-----1",

c,

d

)

)

大功告成!!!!!

類的複製 就不在自己寫了 感覺並不是很難,就直接用原作者的.

類複製說完了物件的複製,我們來看看如何實現類的複製,因為比較簡單,直接放上程式碼定義類複製123456789101112131415#import@interface Person : NSObject@property(strong,nonatomic)NSString *age;

@property(strong,nonatomic)NSString *name;

@end

#import "Person.h"

@implementation Person

- (id)copyWithZone:(NSZone *)zone

{

Person *person = [[Person allocWithZone:zone] init];

person.age = self.age;

person.name = self.name;

return person;

}

@end

呼叫

Person *person = [[Person alloc]init];

person.age = @"dsdsd";

person.name = @"dsdsdddww";

Person *copyPerson = [person copy]; 

NSLog(@"%@-----%@",copyPerson.age, copyPerson.name);可以看到copyPerson的兩個屬性和persona一樣。

@property中的copy關鍵字

在設定NSString型別的屬性的時候,我們最好設定為copy型別,這樣別人使用我們定義的屬性的時候,他不管怎麼改動該屬性的賦值,都不會影響我們給該屬性賦的值,為什麼呢?

下面我們來看看

2491655-31f45b2f9281a420.png

如上圖所示,string2的屬性是copy型別,可以看到是無法被修改的。

因為此時string2和copystring的記憶體地址不一樣,修改一個,不會影響另外一個。

2491655-c47ab5bffc693dec.png

上圖所示,如果string2的屬性是strong型別,就可以被修改,如下圖所示:因為此時string2和copystring的記憶體地址都是一樣的,修改一個,兩個就同時被修改copy關鍵字的NSMutableString崩潰


2491655-087eadd997e7b15a.png

原因:copy關鍵字的string的setter方法實際上是把引數copy之後再賦值給變數_string,那麼此時變數_string雖然被申明為NSMutableString,但是copy之後,就把變數_string變成了不可變的NSString型別,所以就會出現方法報錯,提示對不可變的NSString使用了NSMutableString的方法appendString。

相關文章