内存管理的思考方式
- 自己生成的对象,自己持有
- 非自己生成的对象,自己也能持有
- 不再需要自己持有的对象时释放
- 非自己持有的对象无法释放
ARC 的规则
- 不能使用 retain/release/retainCount/aotorelease
- 不能使用 NSAllocateobject/NSDeallocateObject
- 需遵守内存管理的方法命名规则
- 不要显式调用 delloc
- 使用 @autorelease 块代替 NSAutoreleasePool
- 不能使用区域 NSZone
- 对象变量不能作为C语言结构体(struct/union)的成员
- 显式转换 “id” 和 “void”
修饰词 | 基础数据类型 | MRC | ARC | autorelease |
---|---|---|---|---|
未指定任何类型 | 直接赋值 | 警告 | 警告 | 直接赋值(有可能会提示警告) |
assign | 直接赋值 | 直接赋值 | 直接赋值 | 直接赋值 |
unsafe_unretained | 直接赋值 | 直接赋值 | 直接赋值 | 直接赋值 |
retain | 出错 | 赋值并对新值进行retain操作 | 赋值并对新值进行retain操作 | 直接赋值 |
strong | 出错 | 赋值并对新值进行retain操作 | 赋值并对新值进行retain操作 | 直接赋值 |
weak | 出错 | 直接赋值 | 弱引用 | 直接赋值 |
copy | 出错 | 赋值时键入传入值的一份副本 | 赋值时键入传入值的一份副本 | 赋值时键入传入值的一份副本 |
对 isa , superclass 总结
- instance 的 isa 指向 class
- class 的isa 指向 meta-class
- meta-class 的 isa 指向基类的 meta-class,基类的 meta-class isa 指向自己
- class 的 supercalss 指向父类 的class ,如果 没有父类,superclass 指针为nil
- meta-class 的 superclass 指向父类的 meta-class,基类的 meta-class的 superclass指向基类的 class
- instance 调用方法的轨迹:isa 找到class ,方法不存在就通过superclass 找父类
- class 调用类方法的轨迹 : isa 找 meta-class ,方法不存在,就通过 superclass 找 meta-class的 父类
关联对象
关联对象技术的核心对象:
AssociationsManager
AssociationsHashMap
ObjectAssociationMap
ObjcAssociation
- 一个实例对象就对应一个ObjectAssociationMap,而ObjectAssociationMap里面存储着多个此实例对象的关联对象的key以及ObjcAssociation,ObjcAssociation中存储着关联对象的value和policy。
- 关联对象并不是放在了原来的对象里面,而是自己维护了一个全局的map用来存放每一个对象及其对应关联属性。
- 即使没有调用移除关联对象的方法,在对象销毁的时候,对应的关联对象也会被销毁。因为在dealloc执行的时候,底层会检查是否有关联对象。
iOS编译过程
- 预处理:Clang会预处理你的代码,比如把宏嵌入到对应的位置、注释被删除,条件编译被处理
- 词法分析:词法分析器读入源文件的字符流,将他们组织称有意义的词素(lexeme)序列,对于每个词素,此法分析器产生词法单元(token)作为输出。并且会用Loc来记录位置。
-
语法分析:这一步是把词法分析生成的标记流,解析成一个抽象语法树(abstract syntax tree -- AST),同样地,在这里面每一节点也都标记了其在源码中的位置。
AST 是抽象语法树,结构上比代码更精简,遍历起来更快,所以使用 AST 能够更快速地进行静态检查。 -
静态分析:把源码转化为抽象语法树之后,编译器就可以对这个树进行静态分析处理。静态分析会对代码进行错误检查,如出现方法被调用但是未定义、定义但是未使用的变量等,以此提高代码质量。当然,还可以通过使用 Xcode 自带的静态分析工具(Product -> Analyze)进行手动分析。最后 AST 会生成 IR,IR 是一种更接近机器码的语言,区别在于和平台无关,通过 IR 可以生成多份适合不同平台的机器码。
静态分析的阶段会进行类型检查,比如给属性设置一个与其自身类型不相符的对象,编译器会给出一个可能使用不正确的警告。在此阶段也会检查时候有未使用过的变量等。 - 中间代码生成和优化:此阶段LLVM 会对代码进行编译优化,例如针对全局变量优化、循环优化、尾递归优化等,最后输出汇编代码xx.ll文件。
- 生成汇编代码: 汇编器LLVM会将汇编码转为机器码。此时的代码就是.o文件,即二进制文件。
-
链接:连接器把编译产生的.o文件和(dylib,a,tbd)文件,生成一个mach-o文件。mach-o文件级可执行文件。编译过程全部结束,生成了可执行文件Mach-O
iOS编译过程原文