詳解iOS的深淺拷貝_IOS

來源:互聯網
上載者:User

前言

OC對象的三種拷貝方式

OC的對象拷貝有如下三種方式,很多時候我們把深複製和完全複製混為一談,其他他們是有區別的,具體如下

淺複製(shallow copy):在淺複製操作時,對於被複製對象的每一層都是指標複製。

深複製(one-level-deep copy):在深複製操作時,對於被複製對象,至少有一層是深複製。

完全複製(real-deep copy):在完全複製操作時,對於被複製對象的每一層都是對象複製。

兩圖以避之


理解深複製(mutableCopy)

淺複製很簡單,就不示範了,看上面的圖就懂了,只是簡單的指標拷貝,所以改變原對象或者拷貝後的對象,都會影響另外一個對象。

從上圖我們可以看到mutableCopy對於任何對象都是內容複寫,也就是說進行了深複製。

上代碼:

  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:                [NSMutableString stringWithString:@"1"],                [NSMutableString stringWithString:@"2"],                [NSMutableString stringWithString:@"3"],                [NSMutableString stringWithString:@"4"],                nil                ];  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:                [NSMutableString stringWithString:@"one"],                [NSMutableString stringWithString:@"two"],                [NSMutableString stringWithString:@"three"],                [NSMutableString stringWithString:@"four"],                dataArray1,                nil                ];  NSMutableArray * dataArray3;  NSMutableString * mStr;  dataArray3=[dataArray2 mutableCopy];  mStr = dataArray2[0];  [mStr appendString:@"--ONE"];  NSLog(@"dataArray3:%@",dataArray3);  NSLog(@"dataArray2:%@",dataArray2);

輸出如下:

2016-07-31 17:40:30.702 test1[2113:169774] dataArray3:(  "one--ONE",  two,  three,  four,    (    1,    2,    3,    4  ))2016-07-31 17:40:30.703 test1[2113:169774] dataArray2:(  "one--ONE",  two,  three,  four,    (    1,    2,    3,    4  ))

看上面的輸出,我們發現我們改變原數組dataArray2,竟然也會影響深複製後的dataArray3,不是說好的內容複寫嗎,為什麼會這樣?

這裡我們來說說深複製和完全複製的區別

我們知道深複製,就是把原有對象的內容直接複製一份到新對象,但是這裡有一個坑就是他只會複製一層對象,而不會複製第二層甚至更深層次的對象。

代碼dataArray3=[dataArray2 mutableCopy];只是對數組dataArray2本身進行了內容拷貝,但是裡面的字串對象卻沒有進行內容拷貝,而是進行的淺複製,那麼dataArray2dataArray3裡面的對象是共用同一份的。所以才會出現上面的情況。

單層深複製

那麼如何解決上面的問題呢?

可以使用如下代碼

  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];

輸出如下:

2016-07-31 17:45:48.472 test1[2151:173221] dataArray3:(  one,  two,  three,  four,    (    1,    2,    3,    4  ))2016-07-31 17:45:48.472 test1[2151:173221] dataArray2:(  "one--ONE",  two,  three,  four,    (    1,    2,    3,    4  ))

可以看到dataArray3並沒有被改變,但是別高興的太早,我們再來改改。

代碼如下:

  NSMutableArray * dataArray1=[NSMutableArray arrayWithObjects:                [NSMutableString stringWithString:@"1"],                [NSMutableString stringWithString:@"2"],                [NSMutableString stringWithString:@"3"],                [NSMutableString stringWithString:@"4"],                nil                ];  NSMutableArray * dataArray2=[NSMutableArray arrayWithObjects:                [NSMutableString stringWithString:@"one"],                [NSMutableString stringWithString:@"two"],                [NSMutableString stringWithString:@"three"],                [NSMutableString stringWithString:@"four"],                dataArray1,                nil                ];  NSMutableArray * dataArray3;  NSMutableString * mStr;  dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];  NSMutableArray *mArr = (NSMutableArray *)dataArray2[4];  mStr = mArr[0];  [mStr appendString:@"--ONE"];  NSLog(@"dataArray3:%@",dataArray3);  NSLog(@"dataArray2:%@",dataArray2);

輸出如下:

2016-07-31 17:47:19.421 test1[2174:174714] dataArray3:(  one,  two,  three,  four,    (    "1--ONE",    2,    3,    4  ))2016-07-31 17:47:19.421 test1[2174:174714] dataArray2:(  one,  two,  three,  four,    (    "1--ONE",    2,    3,    4  ))

可以看到深複製又失效了,這是因為dataArray3=[[NSMutableArray alloc]initWithArray:dataArray2 copyItems:YES];僅僅能進行一層深複製,對於第二層或者更多層的就無效了,那怎麼辦呢?

別急,我們還有大招沒放。

完全複製

要想對多層集合對象進行複製,我們需要進行完全複製,這裡可以使用歸檔和接檔。

實現代碼如下:

  dataArray3 = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:dataArray2]];

此時輸出如下:

2016-07-31 17:49:55.561 test1[2202:177163] dataArray3:(  one,  two,  three,  four,    (    1,    2,    3,    4  ))2016-07-31 17:49:55.562 test1[2202:177163] dataArray2:(  one,  two,  three,  four,    (    "1--ONE",    2,    3,    4  ))

可以看到dataArray3沒有被dataArray2的修改影響。

類複製

說完了對象的複製,我們來看看如何?類的複製,因為比較簡單,直接放上代碼

定義類複製

#import <Foundation/Foundation.h>@interface Person : NSObject<NSCopying>@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類型,這樣別人使用我們定義的屬性的時候,他不管怎麼改動該屬性的賦值,都不會影響我們給該屬性賦的值,為什麼呢?

下面我們來看看


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

因為此時string2copystring的記憶體位址不一樣,修改一個,不會影響另外一個。


上圖所示,如果string2的屬性是strong類型,就可以被修改,如下圖所示:

因為此時string2copystring的記憶體位址都是一樣的,修改一個,兩個就同時被修改

copy關鍵字的NSMutableString崩潰

原因:

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

總結

以上就是iOS的深淺拷貝的詳細內容,希望本文在大家開發iOS的過程中能有所協助。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.