浅谈block

原有系统的问题

一个NSInteger的普通变量,它所占用的内存与CPU的位数有关,在32为CPU下占4个字节,在64位下占据8个字节,而指针的类型的大小也通常和CPU位数有关,一个指针所占用在32位CPU下为4个字节,在64位下是8个字节

从32位机器迁移到64位机器中,逻辑不变,但是NSNumber,NSDate一类的对象所占用的内存会翻倍

谈谈效率上的问题,为了存储和访问一个NSNum对象,我们需要在对上为其分配内存,另外还需要维护它的引用计数,管理它的生命周期,这些都是给程序增加额外的逻辑,造成运行效率上的损失

Block的解剖

在objective-C中,一共有三种类型的block

1._NSConcreteStackBlock,保存在栈中的block,

当函数返回时会被销毁

2._NSConcreteMallocBlock,保存在堆中的block

当引用计数器为0时会被销毁

3._NSConcreteGlobalBlock,全局静态block,

不会访问任何外部变量

他们之间的区别可以使用clang-rewrite-objc

block.c命令来查看

详细解说block

1.一个block实际就是一个对象,它主要是由一个isa.一个impl和一个descriptor组成

2.在ARC是,block是_NSConcreteGlobalBlock类.

3.impl是实际函数指针,在本例中它指向_main_blick_func_0.

4.descriptor是用于描述当前这个block的附加信息的,包括结构体的大小,需要capture和dispose的变量列表等.结构体大小需要保存的原因是,每个block会capture一些变量,这些变量会加到_main_block_impl0这个结构体中,使其体积变大.

2.block变量的复制

block外的变量引用,block默认是将其赋值其数据结构中来实现访问的,如果对象是一个引用类型,则block会将其引用计数加1

3.对于__block修饰的外部变量引用,bloc是赋值引用地址来实现访问的.

4.避免循环引用

由于block会赋值外部的变量,所以如果不注意,会造成循环引用.对于这种问题,需要将引用的一方编程weak,从而避免循环引用

你可能感兴趣的:(浅谈block)