iOS_模拟KVO的底层实现、手动实现KVO
一、回顾系统的KVO是怎么实现监听的
#1. 实例化一个类
Person *person = [[Person alloc] init];
person.age = 11;
#2. 开始监听
[person addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
#3. 值改变时(age = 111),监听执行的方法
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
NSLog(@"person 的值变了:%d",_person.age);
}
我们既然要手动写一个KVO,那么就要自己写一个API,让person调喽,还有一个就是值发生变化,也会调一个我们自己的API喽,例如:ml_observer,具体请看下面
二、手动做咱们自己的KVO
核心步骤如下:
一)、生成一个Person类的派生类 Person_KVO。
二)、使当前对象的isa指向新的派生类,就会调用派生类的set方法。
三)、重写Person_KVO的setAge方法,在set方法中拿到观察者(使用运行时关联观察者这个属性)。
四)、每次重写setAge方法,都调用观察者的ml_observeValueForKeyPath方法。实现时刻监听。
代码如下:
ViewController.h
Person *person = [[Person alloc] init];
_person = person;
#调用手写KVO开始监听方法
[person ml_addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
# 监听方法 (验证手写KVO是否成功)
- (void)ml_observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
NSLog(@"person 的值变了:%d",_person.age);
}
NSObject+KVO.h
- (void)ml_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
//修改isa指针,就是把`当前对象`指向一个`新类`
// Person's isa -> Person_KVO
// Class object_setClass(id obj, Class cls)
//使当前对象的isa指向新的派生类(Person_KVO),就会调用派生类的set方法
object_setClass(self, [Person_KVO class]);
//给对象绑定观察者对象
objc_setAssociatedObject(self, @"observer", observer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
Person_KVO.h
-(void)setAge:(int)age{
[super setAge:age]; //修改了子类的值后,父类的值也修改了
//调用KVO
//获取观察者
id observer = objc_getAssociatedObject(self, @"observer");
//调用观察者的方法
//就是验证手写KVO的方法 observer其实就是VC
[observer ml_observeValueForKeyPath:@"age" ofObject:self change:nil context:nil];
}
PS:该代码仅仅是模拟KVO的底层实现流程,主要注重流程,具体的什么参数如观察新值、旧值,没有做,有强迫症的朋友有兴趣可以设置个枚举即可!