關(guān)于 Objective-C 中的 copy

通常, 我們?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è)贊了!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 目標(biāo)近三個(gè)月內(nèi)現(xiàn)金收入增加五萬(wàn)元,幫助更多的人,支助一個(gè)孩子上學(xué)。 1.給父母及男友的父母各存五元錢(qián),感恩他們的付...
    空空dj閱讀 230評(píng)論 0 0
  • 痛快淋漓
    b232349ef6dc閱讀 103評(píng)論 0 1
  • 在噩夢(mèng)雙端 我與你相遇 一方降下悲傷之雨 一方飄零愚者之雪 伴隨審判逼近 裁決已罰于身 瀕臨毀滅時(shí)刻 戰(zhàn)爭(zhēng)將布滿局...
    墨名莫笑閱讀 207評(píng)論 0 0

友情鏈接更多精彩內(nèi)容