OC底层探索15-Strong和Copy区别浅谈

先抛出问题:下方声明的区别是什么?

@property(nonatomic, strong) NSString *strongStr;
@property(nonatomic, copy) NSString *copyStr;

观察下面4个场景

1. NSString场景一

NSString *newString = [NSString stringWithFormat:@"newString"];

_strongStr = newString;
_copyStr = newString;

NSLog(@"newString 对象地址: %p ,对象指针地址:%p ,对象的值:%@", newString, &newString, newString);
NSLog(@"strongStr 对象地址: %p ,对象指针地址: %p ,对象的值:%@", _strongStr, &_strongStr, _strongStr);
NSLog(@"copyStr 对象地址: %p ,对象指针地址:%p ,对象的值:%@", _copyStr, &_copyStr, _copyStr);

输出:



结论:
copy、strong修饰的变量地址都指向newString的内存地址

2. NSString场景二

NSString *newString = [NSString stringWithFormat:@"newString"];

self.strongStr = newString;
self.copyyStr = newString;

输出:



结论:
结论和场景一是相同的

3. NSMutableString场景一

NSMutableString *newString = [NSMutableString stringWithFormat:@"newString"];

_strongStr = newString;
_copyyStr = newString;

[newString setString:@"change String"];

输出:



结论:
结论和场景一、二是相同的

4. NSMutableString场景二

NSMutableString *newString = [NSMutableString stringWithFormat:@"newString"];

self.strongStr = newString;
self.copyyStr = newString;

[newString setString:@"change String"];

NSLog(@"newString 对象地址: %p ,对象指针地址:%p ,对象的值:%@", newString, &newString, newString);
NSLog(@"strongStr 对象地址: %p ,对象指针地址:%p ,对象的值:%@", _strongStr, &_strongStr, _strongStr);
NSLog(@"  copyStr 对象地址: %p ,对象指针地址:%p ,对象的值:%@", _copyyStr, &_copyyStr, _copyyStr);

输出:


结论:
copy修饰的变量,对象地址不一致了,指针指向了一个新的内存区域(相当于深拷贝),导致新值(newString)修改时不会影响。copy修饰符到底做了什么?这就是我们探索的起点

接下来一步一步解释:

self.strongStr&_strongStr两种方式的区别

这个相信结论大家都是知道的:

  • self.strongStr这种方式是调用了set方法;
  • _strongStr种方式是直接对值进行修改;

通过clang来查看strongStr变量的两种不同写法编译后的源码:

  • self + OBJC_IVAR_...(属性偏移值) = strongStr的内存地址,然后在内存中进行替换。

所以在日常使用时,建议多使用_strongStr这种方式.(虽然性能提升的非常有限,但态度要有【狗头】)

通过clang来查看copyStr变量的两种不同写法编译后的源码:

结论:
观察下来使用copy或strong对于编译后的源码并没有发现什么本质的区别,那问题一定是出在set方法上。

copy&strong导致set方法不同

  • strong修饰变量的Set方法


    结论:
    strong修饰变量的set方法和直接赋值都是:通过指针偏移后,将变量指针指向新的地址。

  • copy修饰变量的Set方法


    结论:
    copy修饰变量的直接赋值是:通过指针偏移后,将变量指针指向新的地址。而set方法则调用的objc_setProperty函数,问题一定出在这个函数上。

objc_setProperty函数

1. 在llvm中搜索objc_setProperty

为什么copy修饰的变量set方法是调用objc_setProperty函数,而strong修饰却没有呢?因为苹果在llvm中对set方法做了处理.

2. 然后再objc4-818的源码中搜索objc_setProperty_nonatomic_copy
3. 在源码中增加断点,继续深入


在这个位置发现了关键,使用copy修饰属性之后。属性的set方法是调用了新值的copy协议,也就是调用了NSMutableString的copyWithZone方法

4. NSMutableString的copyWithZone方法

Founation苹果并没有开源,所以需要别的途径。其中有几个思路:CFFounation、Swift中的Founation(开源)、GNUstep。其中:CFFounation根本就没有满足NSCopying协议;Swift虽然开源了,但是不够明确。最终发现了GNUstep-翀鹰精灵。然后我打开了新世界。

  • NSMutableString并没有找到对应的copyWithZone,继续向上找到父类NSStringcopyWithZone
  • 发现最终调用了NSMutableStringallocWithZone
5. NSMutableString的allocWithZone
6. NSAllocateObject方法
  • 到这就可以得出结论了,NSMutablString的Copy协议是创建了新的内存空间,进行了内容拷贝,通俗可以理解为进行了深拷贝
7. 最后一步initWithString方法

结论:

  • 场景3、4时,通过copy修饰的NSString、NSArray、NSDictory类型变量,在进行Set方法时,会调用objc_setProperty函数,而最终会调用新值对应类型(NSMutableString)copyWithZone。通过第6步可知,就是完成了一次深拷贝,从而生成了一个新的对象,并且copy的对象指向这个新对象;
  • 场景1、2时,新值的类型是NSString,在copy时进行了浅拷贝

一般声明不可变类型,就是不希望它变化,所以还是建议使用Copy来修饰,虽然浪费了内存但是更加安全。

补充

assign修饰的变量Set方法
  • assign的Set方法也是通过地址偏移来完成值的修改。
使用atomic一定是线程安全的吗?

atomic可以保证setter和getter存取的线程安全并不保证整个对象是线程安全的。
比如,声明一个NSMutableArray的原子属性array,此时self.array和self.array = otherArray都是线程安全的。但是,使用[self.array objectAtIndex:index]就不是线程安全的,需要用锁来保证线程安全性。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容