Block探索
Block
block 类型
**1、全局block - **NSGlobalBlock****
**2、栈block -NSStackBlock **
**3、堆block - **NSMallocBlock****
总结
-
block直接存储在全局区
-
如果block访问外界变量,并进行block相应拷贝,即copy
-
如果此时的block是强引用,则block存储在堆区,即堆区block
-
如果此时的block通过__weak变成了弱引用,则block存储在栈区,即栈区block
解决block循环引用的方法 自动释放 如果block内部嵌套block,需要同时使用__weak 和 **strong image.jpeg
虽然**weak,解决了block里面的循环引用问题,但是在延迟操作的时候,保证不了name的生命周期,所以这时候是null,这时候就要结合__strong 来使用
weak-strong-dance 强弱共舞
手动释放
利用临时变量打破block->self的持有
利用VC进行通讯
NSProxy 虚拟类
NSProxy 其实是一个消息重定向封装的一个抽象类,类似一个代理人,中间件,可以通过继承它,并重写下面两个方法来实现消息转发到另一个实例
循环引用解决原理
自定义一个NSProxy的子类CJLProxy
block底层分析
通过xcrun -sdk iphonesimulator clang -arch x86_64 -rewrite-objc block.c,将block.c 编译成 block.cpp,其中block在底层被编译成了以下的形式
image.jpeg image.jpeg image.jpeg
这里说明了block必须调用
访问外界局部变量
image.jpeg image.jpeg
这里会造成编译器代码歧义,只读
__block修饰后
image.jpeg image.jpeg image.jpeg
block的结构
Block_layout
descriptor1,通过指针编译,可以获取到descriptor2,copy和dipose在这里
descriptor3,block的签名在这里
block变化的流程
image.jpeg开始是一个全局的block,但是访问了外部局部变量,变成栈block,之后在系统内部,objc_retainblock里面,系统进行了copy的操作,外部默认是strong修饰的,所以出objc_retainblock的时候,就变成了堆block
_Block_copy
把参数转成block_layout类型,如果是全局block那么直接返回,再之后只有栈区block了,因为堆区block是需要申请开辟空间的,进入栈的判断,这里进行了malloc申请开辟空间,进行memmove平移,copy变成堆区block
block的三层copy
1、首先blcok对外部变量的一次copy,到堆区
2、因为block修饰的外部变量变成了_block_byref的结构体,这一次对这个结构体进行copy
3、如果外部变量是一个对象的话,那么就会进行第三次copy,对这个对象进行copy到block内部
blcok对外界变量的捕获
首先判断对象类型,byref是block修饰后的类型