首先ARC之前属性构造器的关键字是retain,copy,assign,strong和weak是ARC之后引入的关键字。最新写法一般也不推荐使用 retain了。
首先retain和strong几乎等同
声明属性时用strong或者retain效果是一样的。
retain和strong的区别在于修饰block的区别。我们都知道block应该使用copy进行修饰。先说结果:
在MRC(Manual Reference Counting)下
修饰block时,strong相当于copy。arc下
修饰block时,retain相当于assign。(栈block引用计数增加了,但是栈block捕获的变量没有增加也没有copy复制到堆上还是在原来的栈区已经被释放了,成野指针了 ,block还有block捕获的局部变量没了)
导致结果:如果用strong修饰没有问题,如果用retain修饰会崩溃。报野指针错误。
详解:
在MRC下,block的存储分为三块:堆区,栈区,全局区。在ARC下,block的存储分为两块:全局区,堆区。
如果在MRC下,我们用全局变量对栈区的block进行了引用,然后修饰符还是使用的retain或者assign,那么结果就是后边使用可能出现野指针访问坏内存。
结论就是block使用strong或者copy进行修饰最安全。
作者:frankisbaby
链接:https://www.jianshu.com/p/e2ccf2ae2b98
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
weak和assign
都可以打破循环引用
weak只能用来修饰对象 安全的把指针指向nil
assign既可以是对象也可以是基本类型 指向对线可能不安全(已释放或者已更改)
copy和strong copy和retain
参见 https://www.jianshu.com/p/88b012f2c803
实际应用中在修饰不可变对象 array等时没区别 可变对象会出问题
因为字符串 @"aaa"的特殊性 视为字面量常量性能等问题 所以一般用copy来修饰
但是NSMutableString 要用strong 用copy会深拷贝成不可变字符串
__weak和__block
1.__block不管是ARC还是MRC模式下都可以使用,可以修饰对象,还可以修饰基本数据类型。
2.__weak只能在ARC模式下使用,也只能修饰对象(NSString),不能修饰基本数据类型(int)。
3.__block对象可以在block中被重新赋值,__weak不可以。
4.__block对象在ARC下可能会导致循环引用,非ARC下会避免循环引用,__weak只在ARC下使用,可以避免循环引用。
1:weak
很少有人知道weak表其实是一个hash(哈希)表,Key是所指对象的地址,Value是weak指针的地址数组。更多人的人只是知道weak是弱引用,所引用对象的计数器不会加一,并在引用对象被释放的时候自动被设置为nil。通常用于解决循环引用问题。但现在单知道这些已经不足以应对面试了,好多公司会问weak的原理。weak的原理是什么呢?下面就分析一下weak的工作原理(只是自己对这个问题好奇,学习过程中的笔记,希望对读者也有所帮助)。
weak 实现原理的概括
Runtime维护了一个weak表,用于存储指向某个对象的所有weak指针。weak表其实是一个hash(哈希)表,Key是所指对象的地址,Value是weak指针的地址(这个地址的值是所指对象的地址)数组。
weak 的实现原理可以概括一下三步:
1、初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。
2、添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。
3、释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。
3:strong
默认用strong进行修饰 本质是引用计数+1 维护的该变量的引用计数哈希表
//Assigning retained object to weak variable; object will be released after assignment
__weak UIlabel * _weakLabel = [[UILabel alloc] initWithFrame:CGRectMake(30, 170, 200, 60)];
_weakLabel.backgroundColor = [UIColor cyanColor];
_weakLabel.text = @"我是weak";
_weakLabel.textAlignment = NSTextAlignmentCenter;
[self.view addSubview:_weakLabel];
初始化后对象因为是弱引用引用计数不增加 对象释放 __weak指向nil;
UILabel *l1 = [[UILabel alloc] initWithFrame:CGRectMake(30, 170, 200, 60)];
__weak UILabel * _weakLabel = l1;
_weakLabel.backgroundColor = [UIColor cyanColor];
_weakLabel.text = @"我是weak";
_weakLabel.textAlignment = NSTextAlignmentCenter;
[self.view addSubview:_weakLabel];