Objective-C是一门动态性比较强的编程语言,跟C、C++等语言有着很大的不同,Object-C的函数调用不是在编译时期决定的,而是在运行时决定的。Objective-C的动态性是由Runtime API来支撑的,Runtime API提供的接口基本都是C语言的,源码由C\C++\汇编语言编写。
isa详解
在arm64架构之前,isa就是一个普通的指针,存储着Class、Meta-Class对象的内存地址,从arm64架构开始,对isa进行了优化,变成了一个共用体(union)结构,还使用位域来存储更多的信息
isa位域
nonpointer
0,代表普通的指针,存储着Class、Meta-Class对象的内存地址
1,代表优化过,使用位域存储更多的信息
has_assoc
是否有设置过关联对象,如果没有,释放时会更快
has_cxx_dtor
是否有C++的析构函数(.cxx_destruct),如果没有,释放时会更快
shiftcls
存储着Class、Meta-Class对象的内存地址信息
magic
用于在调试时分辨对象是否未完成初始化
weakly_referenced
是否有被弱引用指向过,如果没有,释放时会更快
deallocating
对象是否正在释放
extra_rc
里面存储的值是引用计数器减1
has_sidetable_rc
引用计数器是否过大无法存储在isa中
如果为1,那么引用计数会存储在一个叫SideTable的类的属性中
Class的结构
Class的本质是一个objc_class的结构体,结构体内包含指向原类的Class isa
指针,指向父类的Class superclass
指针,缓存方法列表cache_t cache
,类信息class_data_bits_t bits
,bits & FAST_DATA_MASK
得到class_rw_t
结构体,class_rw_t结构体包含method_list_t * methods
方法列表 property_list_t * properties
属性列表 const protocol_list_t * protocols
等信息,还包含一个特殊的结构体指针就是const class_ro_t
class_ro_t是类最原始的类信息,是只读的结构体,包含最开始的类名、成员变量类标、方法列表、协议列表等信息。
初始化class_rw_t结构体时会将class_ro_t结构体里的方法列表和协议列表等信息复制到class_rw_t结构体内,通过runtime添加的属性和方法也都是添加到class_rw_t结构体内。
class_rw_t里面的methods、properties、protocols是二维数组,是可读可写的,包含了类的初始内容、分类的内容
class_ro_t里面的baseMethodList、baseProtocols、ivars、baseProperties是一维数组,是只读的,包含了类的初始内容
method_t结构体是对方法\函数的封装,包含SEL name
const char * types
IMP imp
- IMP代表函数的具体实现
- SEL代表方法\函数名,一般叫做选择器,底层结构跟char *类似可以通过@selector()和sel_registerName()获得,可以通过sel_getName()和NSStringFromSelector()转成字符串,不同类中相同名字的方法,所对应的方法选择器是相同的
- types包含了函数返回值、参数编码字符串