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 *)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 *)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的底层实现流程,主要注重流程,具体的什么参数如观察新值、旧值,没有做,有强迫症的朋友有兴趣可以设置个枚举即可!

三、流程图

iOS_模拟KVO的底层实现、手动实现KVO_第1张图片
795875-63631d06f874e17b.png.jpeg

你可能感兴趣的:(iOS_模拟KVO的底层实现、手动实现KVO)