原有系统的问题
一个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,从而避免循环引用