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 强弱共舞

手动释放

image.jpeg

利用临时变量打破block->self的持有

image.jpeg


利用VC进行通讯
NSProxy 虚拟类
NSProxy 其实是一个消息重定向封装的一个抽象类,类似一个代理人,中间件,可以通过继承它,并重写下面两个方法来实现消息转发到另一个实例

image.jpeg

循环引用解决原理
自定义一个NSProxy的子类CJLProxy

block底层分析

image.jpeg


通过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

image.jpeg

descriptor1,通过指针编译,可以获取到descriptor2,copy和dipose在这里
descriptor3,block的签名在这里

block变化的流程

image.jpeg

开始是一个全局的block,但是访问了外部局部变量,变成栈block,之后在系统内部,objc_retainblock里面,系统进行了copy的操作,外部默认是strong修饰的,所以出objc_retainblock的时候,就变成了堆block

_Block_copy

image.jpeg

把参数转成block_layout类型,如果是全局block那么直接返回,再之后只有栈区block了,因为堆区block是需要申请开辟空间的,进入栈的判断,这里进行了malloc申请开辟空间,进行memmove平移,copy变成堆区block

block的三层copy
1、首先blcok对外部变量的一次copy,到堆区
2、因为block修饰的外部变量变成了_block_byref的结构体,这一次对这个结构体进行copy
3、如果外部变量是一个对象的话,那么就会进行第三次copy,对这个对象进行copy到block内部

image.jpeg

blcok对外界变量的捕获

image.jpeg

image.jpeg


首先判断对象类型,byref是block修饰后的类型

image.jpeg

image.jpeg

你可能感兴趣的:(Block探索)