iOS-知识点整理(不定时更新)

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: 方法时,方法内部会做以下操作:

  1. 判断有没有指定key的set方法,如果有set方法,就会调用set方法,给该属性赋值
  2. 如果没有set方法,判断有没有跟key值相同且带有下划线的成员属性(_key).如果有,直接给该成员属性进行赋值
  3. 如果没有成员属性_key,判断有没有跟key相同名称的属性.如果有,直接给该属性进行赋值
  4. 如果都没有,就会调用 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;
    };
}

你可能感兴趣的:(iOS-知识点整理(不定时更新))