淺拷貝:拷貝容易本身,返回一個(gè)對(duì)象,指向相同的內(nèi)存地址.
深層復(fù)制:拷貝容器本身,返回一個(gè)對(duì)象,指向不同的內(nèi)存地址.
例如NSArray、NSDictionary,它們已經(jīng)實(shí)現(xiàn)了上面兩個(gè)協(xié)議。
對(duì)于它們來(lái)說(shuō),規(guī)則很簡(jiǎn)單,obj2 = [obj1 copy]返回的必然是一個(gè)不可變對(duì)象,無(wú)論obj1是可變對(duì)象還是不可變對(duì)象。如果obj1是一個(gè)不可變對(duì)象,那么它們指向同一個(gè)對(duì)象,也是上一條我提到過(guò)的。
obj2 = [obj1 mutableCopy]返回的必然是一個(gè)可變對(duì)象,無(wú)論obj1是可變對(duì)象還是不可變對(duì)象。即使obj1也是一個(gè)可變對(duì)象,它們?nèi)灾赶虿煌刂?,是兩個(gè)對(duì)象。

注意:其他對(duì)象NSArray、NSMutableArray 、NSDictionary、NSMutableDictionary一樣適用
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy]; # 這里崩潰
看崩潰信息HSPerson應(yīng)該先實(shí)現(xiàn):
- (id)copyWithZone:(NSZone *)zone;
測(cè)試:
#import "HSPerson.h"
@interface HSPerson()<NSCopying>
@end
@implementation HSPerson
- (id)copyWithZone:(NSZone *)zone
{
return @"漢斯哈哈哈";
}
@end
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy];
NSLog(@"copyP: %@", copyP);
可以看出copyWithZone重新分配新的內(nèi)存空間,則:
- (id)copyWithZone:(NSZone *)zone
{
HSPerson *person = [[HSPerson allocWithZone:zone] init];
return person;
// 有些人可能下面alloc,重新初始化空間,但這方法已給你分配了zone,自己就無(wú)需再次alloc內(nèi)存空間了
// HSPerson *person = [[HSPerson alloc] init];
}
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy];
NSLog(@"p = %p copyP = %p", p, copyP);
NSLog(@"age = %d height = %f", copyP.age, copyP.height);
雖然copy了份新的對(duì)象,然而age,height值并未copy,那么:
- (id)copyWithZone:(NSZone *)zone
{
HSPerson *person = [[HSPerson allocWithZone:zone] init];
person.age = self.age;
person.height = self.height;
// 這里self其實(shí)就要被copy的那個(gè)對(duì)象,很顯然要自己賦值給新對(duì)象,所以這里可以控制copy的屬性
return person;
}
這時(shí)你會(huì)想,有NSMutableCopying?沒(méi)錯(cuò),是有這貨:
- (id)mutableCopyWithZone:(NSZone *)zone
{
HSPerson *person = [[HSPerson allocWithZone:zone] init];
person.age = self.age;
person.height = self.height;
return person;
}
NSCopying、NSMutableCopying有啥區(qū)別?
其實(shí)感覺(jué)沒(méi)必要有NSMutableCopying,因?yàn)閴焊蜎](méi)可變的HSPerson,但如果該對(duì)象有其他行為,可以借用NSMutableCopying實(shí)現(xiàn),哈哈哈
property里的copy、strong區(qū)別
說(shuō)完深淺拷貝,理解copy.strong就輕松多了!
copy
#import <Foundation/Foundation.h>
@interface HSPerson : NSObject
@property (nonatomic, copy) NSString *name;
@end
NSMutableString *string = [NSMutableString stringWithFormat:@"漢斯哈哈哈"];
HSPerson *person = [[HSPerson alloc] init];
person.name = string;
// 不能改變person.name的值,因?yàn)槠鋬?nèi)部copy新的對(duì)象
[string appendString:@" hans"];
NSLog(@"name = %@", person.name);
property copy 實(shí)際上就對(duì)name干了這個(gè):
- (void)setName:(NSString *)name
{
_name = [name copy];
}
假設(shè)name為NSMutableString,會(huì)發(fā)生什么事?
@property (nonatomic, copy) NSMutableString *name;
這樣會(huì)挨罵哦,實(shí)際上內(nèi)部還是:
- (void)setName:(NSMutableString *)name
{
_name = [name copy];
}
copy出來(lái)的仍然是不可變字符!如果有人用NSMutableString的方法,就會(huì)崩潰:
strong
@property (nonatomic, strong) NSString *name;
NSMutableString *string = [NSMutableString stringWithFormat:@"漢斯哈哈哈"];
HSPerson *person = [[HSPerson alloc] init];
person.name = string;
// 可以改變person.name的值,因?yàn)槠鋬?nèi)部沒(méi)有生成新的對(duì)象
[string appendString:@" hans"];
NSLog(@"name = %@", person.name);
總結(jié):用copy與strong取決于需求,如果不希望被外界更改用copy,反之用strong