什么是引用计数?
- 当我们创建一个新对象时,它的引用计数为1
- 当有一个新的指针指向这个对象时,我们将引用计数加1
- 当某个指针不再指向这个对象时,我们将引用计数减1
- 当对象的引用计数为0时,说明这个对象不再被任何指针指向了,就可以将对象销毁,回收内存
循环引用问题
A的销毁依赖于B的销毁,同样B的销毁依赖于A的销毁
-
block
block是将函数及其执行上下文封装起来的==对象==
在 Objective-C 语言中,一共有 3 种类型的 block:
- _NSConcreteGlobalBlock 全局的静态 block,不会访问外部局部变量(显然包括无外部变量或者全局变量)。
- _NSConcreteStackBlock 保存在栈中的 block,当函数返回时会被销毁。
- _NSConcreteMallocBlock 保存在堆中的 block,当引用计数为 0 时会被销毁。
block截获数据
- 对基本数据类型,截获其值
- 对于局部变量或成员变量截获连同所有权修饰符(strong)
- 对于局部静态变量,以指针形式截获
- 对于全局变量(基本数据类型)和全局静态变量不截获
==block内部为什么要用strong==
在block内部的weakSelf有可能为为self或者为nil (比如当前界面正在加载网络数据, 而此时用户关闭了该界面). 这样在某些情况下代码会崩溃. 所以为了让self不为nil, 我们在block内部将weakSelf转成strongSelf. 当block结束时, 该strongSelf变量也会被自动释放. 既避免了循环引用, 又让self在block内部不为nil.
默认情况下,block是存档在栈中,可能被随时回收,通过copy操作可以使其在堆中保留一份, 相当于一直强引用着
==什么时候栈上的Block会复制到堆呢?==
- 调用Block的copy实例方法时
- Block作为函数返回值返回时
- 将Block赋值给附有__strong修饰符id类型的类或者Block类型成员变量时
- 在方法名中含有usingBlock的Cocoa框架方法或Grand Central Dispatch 的API中传递Block时
==__block 和 __weak==
- 1,__block会持有该对象,即使超出了该对象的作用域,该对象还是会存在的,直到block对象从堆上销毁;而__weak仅仅是将该对象赋值给weak对象,当该对象销毁时,weak对象将指向nil;
- 2,_block可以让block修改局部变量(赋值而不是使用,比如给数组添加元素不需要__block),而__weak不能。
- 3,MRC中__block是不会引起retain;但在ARC中__block则会引起retain。所以ARC中应该使用__weak。
- 4,__block 在编译后使局部基本数据类型变为_Block前缀命名的结构体对象,包含isa指针,所以修改的是其地址,而不是值。
==为什么__block可以修改外部变量==
当外部变量没有使用__block ,block 会创建一个新的变量val来保存所截获的外部变量瞬时值,新的val将成为block内的成员变量,之后再代码中修改的是成员变量val值,而不是外部变量,所以外部变量不会受到影响。
而使用了__block之后,保存了外部对象的引用,在block内部修改的是内存。
代码表示:
__block int var = 1;
void block() {
int *ptr = &var;
*ptr++;
}
runtime如何实现weak变量的自动置nil?
weak 的实现原理可以概括一下三步:
- 初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。
- 添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。weak表其实是一个hash,Key是所指对象的地址,Value是weak指针的地址,实现为一个weak_table_t结构体。
- 释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。
变成nil时机
对象被释放的时候, 其dealloc方法执行之前, 它的所有weak属性都已经被设置为nil. 因此, 如果期望在dealloc里访问weak属性, 那是不行的.
使用__weak 修饰符的变量,即是使用注册到autoreleasepool中的对象
- objc_loadWeakRetained函数取出附有__weak修饰符变量所引用的对象并retain
- objc_autorelease函数将对象注册到autorelease中。
weak引用只会在autoreleasepool drain的时候才会被更新为nil。
不用strong->weak的方式来避免循环引用
- 把自己当参数(FaceBook)
typedef BOOL (^POPCustomAnimationBlock)(id target, POPCustomAnimation *animation);- 你总是可以得到一个这些参数的隐含引用(通过block的局部变量获取)所以换句话说,block中的参数是多余的,但是它非常有用,因为现在你可以用这些参数而不是在block外声明一个weak引用在用它
- self不直接或间接持有该Block(Masonry)
- 完成后手动释放Block(AFNetworking)
- 类方法并没有直接或间接持有Block(系统的类方法 UIview animation) animation framework -> block block -> self
GCD -> block block -> self