通常, 我們?cè)诙x 不可變字符串和不可變集合 的時(shí)候, 會(huì)用到 copy 這個(gè)存儲(chǔ)屬性, 那我們?yōu)楹我?copy? copy 到底給我們帶來(lái)了什么?
要回答這個(gè)問(wèn)題, 首先來(lái)了解一下 copy
什么是 copy
如果你有 Objective-C 的編程基礎(chǔ), 那么你一定知道 Strong(強(qiáng)引用), 和 weak(弱引用), 那么 copy 是什么呢?
總所周知, 你定義一個(gè) strong 或是 weak 屬性的時(shí)候, 默認(rèn)的都會(huì)創(chuàng)建以一個(gè)下劃線( _ )開(kāi)頭的同名實(shí)例變量, 強(qiáng)弱引用也是和定義的時(shí)候保持一致, 那么 copy 呢?
當(dāng)然, 沒(méi)有 copy 這種引用關(guān)系, 那到底是什么呢? 我們來(lái)實(shí)際測(cè)試一下
@interface ViewController ()
@property (nonatomic, copy) NSArray *arrayTest;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSArray *array = [[NSArray alloc] initWithObjects:@"1", @"2", nil];
_arrayTest = array;
array = nil;
NSLog(@"%@", _arrayTest.lastObject);
}
@end
上面這段代碼中, 我們直接使用 arrayTest 這個(gè)屬性的實(shí)例變量, 僅僅只是想確認(rèn)實(shí)例變量的存儲(chǔ)屬性是 strong, weak, 還是 unsafe unretain
可以看到, 如果是 weak, 那么打印出來(lái)應(yīng)該是 (null), 如果是 unsafe unretain 那么會(huì)崩潰, 如果是 strong, 那么就會(huì)正常打印出 2 這個(gè)元素
實(shí)際結(jié)果如何?

看來(lái)是強(qiáng)引用.
所以, 用 copy 定義的屬性, 對(duì)應(yīng)的是一個(gè)強(qiáng)引用的實(shí)例變量
copy 做了什么
copy 對(duì)應(yīng)的 setter 中, 會(huì)調(diào)用 copy 方法
比如
-(void)setArrayTest:(NSArray *)arrayTest{
_arrayTest = [arrayTest copy];
}
[arrayTest copy] 會(huì)調(diào)用 copyWithZone: 方法, 后者是 NSCopying 協(xié)議的一個(gè)方法, 作用就是返回一個(gè)自己的拷貝.
需要注意的是, 返回的拷貝對(duì)象是不可變的對(duì)象, 例如, NSMutableArray copy 后會(huì)返回 NSArray 對(duì)象.
另外, 對(duì)于不可變對(duì)象, copy 其實(shí)返回的就是自身, 并沒(méi)有拷貝.
下面寫(xiě)個(gè)代碼測(cè)試一下
NSMutableArray *mutableArray = [[NSMutableArray alloc] initWithObjects:@"1", @"2", nil];
NSArray *mutableArrayCopy = [mutableArray copy];
[mutableArray addObject:@"3"];
NSLog(@"%d", [mutableArrayCopy isKindOfClass:[NSMutableArray class]]);
NSLog(@"%p, %p", mutableArrayCopy, mutableArray);
NSLog(@"%@, %@", mutableArrayCopy, mutableArray);
NSArray *array = [[NSArray alloc] initWithObjects:@"1", @"2", nil];
NSArray *arrayCopy = [array copy];
NSLog(@"%p, %p", array, arrayCopy);

所以, copy 做的就是針對(duì)可變對(duì)象, 生成一份不可變對(duì)象, 如果本身就是不可變對(duì)象, 則返回自己.
為什么要用 copy
使用 copy 主要是為了保證數(shù)據(jù)一致性.
例如, 想象一下, 你自己寫(xiě)了一個(gè)類似 UILabel 的控件, MyLabel, 里面有一個(gè)屬性text, 使用的是 strong 修飾
@property(nonatomic, strong) NSString *text;
在 setter 里面, 為了減少不必要的重繪, 做了一點(diǎn)優(yōu)化, 自以為完美無(wú)缺, 甚至想要給自己點(diǎn)個(gè)贊!
-(void)setText:(NSString *)text{
if( _text == text) {
return;
}
_text = text;
[self setNeedsDisplay];
}
但是如果碰到某個(gè)奇怪的開(kāi)發(fā)者, 寫(xiě)上這個(gè)代碼
- (void)viewDidLoad {
[super viewDidLoad];
self.titleString =[NSMutableString stringWithString:@"12345"];
self.titleLabel.text = self.titleString;
}
然后在某個(gè)事件響應(yīng)里面寫(xiě)上
[self.titleString appendString:@"678"];
self.titleLabel.text = self.titleString;
那么界面上永遠(yuǎn)也顯示不出來(lái)后面加上的這一段字符.
有一個(gè)修改辦法就是把 MyLabel 中的 text 屬性改成 copy
然后 setter 中也改成
_text = [text copy];
這樣就搞定了! 既沒(méi)有多余的重繪, 也沒(méi)有漏掉的重繪! 這次可以點(diǎn)個(gè)贊了!