Bolck
- 在 ARC 中编译器会自动对 block 进行 copy 操作,但是苹果仍然建议使用 copy 来指明编译器的行为
- block 在捕获外部变量的时候,会保持一个强引用,当在 block 中捕获 self 时,由于对象会对 block 进行 copy,于是便形成了强引用循环
- 多次调用 weakSelf 的方法,有可能在 block 执行过程中 weakSelf 变为 nil,
涉及到 weak 本身的机制了。weak 置 nil 的操作发生在 dealloc 中
最后一个持有 object 的对象被释放的时候,会触发对象的 dealloc,而这个持有者的释放操作就不一定保证发生在哪个线程了。因此 block 执行的过程中 weakSelf 有可能在另外的线程中被置为 nil。
- MRC 下Block 默认是分配在栈上的,除非进行显式的 copy
- ARC 也更加倾向于把 Block 放到堆上
Block进阶
- Block在内存中的位置分为三种类型NSGlobalBlock,NSStackBlock, NSMallocBlock
参考资料http://blog.devtang.com/2013/07/28/a-look-inside-blocks/
4 堆和栈
- 堆内存要程序员手动回收(动态分配和回收内存的)
- 栈内存会被系统自动回收(静态分配和动态分配)
静态分配是系统编译器完成的,比如局部变量的分配
动态分配是由alloc函数进行分配的,但是栈的动态分配和堆是不同的,它的动态分配也由系统编译器进行释放,不需要程序员手动管理
- 只有oc对象需要进行内存管理
- 非oc对象类型比如基本数据类型不需要进行内存管理
- 栈是向低地址扩展的数据结构,是一块连续的内存的区域。
堆是向高地址扩展的数据结构,是不连续的内存区域。(堆是链表方式存储)
两者区别
- 管理方式不同;(栈:自动 堆:手动)
- 空间大小不同;( 栈:1M?? 32位系统 堆4G )
- 能否产生碎片不同;(栈:后进先出 堆:频繁new release 大量的碎片使程序效率变低)
- 生长方向不同;(栈: 生长方向是向下的,是向着内存地址减小的方向增长 堆:相反)
- 分配方式不同;(栈:静态和动态 堆:全动态)
- 分配效率不同;(栈:机器系统提供的数据结构,计算机会在底层对栈提供支持:分配专门的寄存器存放栈的地址,压栈出栈都有专门的指令执行,这就决定了栈的效率比较高. 堆:C/C++函数库提供的,它的机制是很复杂的,例如为了分配一块内存,库函数会按照一定的算法(具体的算法可以参考数据结构/操作系统)在堆 内存中搜索可用的足够大小的空间,如果没有足够大小的空间(可能是由于内存碎片太多),就有可能调用系统功能去增加程序数据段的内存空间,这样就有机会分 到足够大小的内存,然后进行返回。显然,堆的效率比栈要低得多。)