runtime
1. 消息传递
先到缓存查找IMP如果没有,则根据isa指针找到类,然后再去methodlist查找有没有方法,
如果没有,再去父类里面查找,
如果再没有,进入消息转发
2. 消息转发
总共有3次机会挽救
- 动态方法解析
Objective-C运行时会调用 +resolveInstanceMethod:或者 +resolveClassMethod:,让你有机会提供一个函数实现。
如果你添加了函数并返回YES, 那运行时系统就会重新启动一次消息发送的过程。 - 备用接收者
- 完整消息转发
3. runtime运用
字典转模型,访问ivar_list进行字典转模型
防奔溃:imgname swizzle 方法交换
4. runtime结构体
isa指针
superclass 指针指向父类
ivars 成员列表
methodlist 方法列表
Cache 缓存列表,哈希
protocols 协议列表
KVC/KVO
Kvc主要是给成员属性赋值的,能够对私有成员变量赋值
底层原理
当一个对象调用setValue:forKey: 方法时,方法内部会做以下操作:
- 判断有没有指定key的set方法,如果有set方法,就会调用set方法,给该属性赋值
- 如果没有set方法,判断有没有跟key值相同且带有下划线的成员属性(_key).如果有,直接给该成员属性进行赋值
- 如果没有成员属性_key,判断有没有跟key相同名称的属性.如果有,直接给该属性进行赋值
- 如果都没有,就会调用 valueforUndefinedKey 和setValue:forUndefinedKey:方法
具体例子:字典转模型
Kvo监听某个对象属性值的改变
底层实现原理
1.KVO 是基于 runtime 机制实现的
2.当一个对象(假设是person对象,对应的类为 JLperson)的属性值age发生改变时,系统会自动生成一个继承自JLperson的类NSKVONotifying_JLPerson,在这个类的 setAge 方法里面调用
[super setAge:age];
[self willChangeValueForKey:@"age"];
[self didChangeValueForKey:@"age"];
具体例子:监听键盘高度
代理/通知/block
1.代理
属于一对一,协议拥有控制链关系(has-a)
优点: 相对而言比较详细
缺点: 这个类必须支持委托,某一时间只能有一个委托连接到某一对象
使用场景: 反向传值
注意点:必须weak,注意循环引用
- weak:指明该对象并不负责保持delegate这个对象,delegate这个对象的销毁由外部控制
- strong:该对象强引用delegate,外界不能销毁delegate对象,会导致循环引用(Retain Cycles)
2.通知
- 属于多对多的,需要有NSNotificationCenter(通知中心)
- 使用场景: 键盘高度监听
- 优点:双方都不需要知道接受者是谁
- 注意点:结束后必须移除通知
3.block
- block本质上也是一个OC对象,它内部也有个isa指针
- block是封装了函数调用以及函数调用环境的OC对象
- block是封装函数及其上下文的OC对象
3.1 变量捕获
- 可以捕获但是不可以修改局部变量
- 可以捕获并且修改
全局变量、静态变量(实际为指针地址,故可以修改)、_block修饰的局部变量
3.2 为什么要用copy
定义 Block:A,B,C
- 如果没有访问外部变量,那这个block-A成为全局的静态block
- 如果没用copy修饰而且去访问了外部变量,block-B会被保存在栈里,会在函数调用结束的时候销毁
- 如果用copy修饰而且去访问了外部变量,block-C会跟随对象销毁而销毁,对象不销毁,堆中的block就可以被我们调用
3.3 循环引用经典例子
typedef void(^block)();
@property (copy, nonatomic) block myBlock;
@property (copy, nonatomic) NSString *blockString;
- (void)MineBlock {
self.myBlock = ^() {
NSString *localString = self.blockString;
};
}
网友评论