KVO

Apple 使用了 isa 混写(isa-swizzling)来实现 KVO 。当观察对象A时,KVO机制动态创建一个新的名为:NSKVONotifying_A 的新类,该类继承自对象A的本类,且 KVO 为 NSKVONotifying_A 重写观察属性的 setter 方法,setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象属性值的更改情况。

自己实现KVO

由于KVO要对所有的NSObject都可以使用,所以创建NSObject的Category

NSObject+NNKVO.h

#import <Foundation/Foundation.h>
@interface NSObject (NNKVO)
    - (void)NN_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
@end

NSObject+NNKVO.m

#import "NSObject+NNKVO.h"
#import <objc/message.h>
@implementation NSObject (NNKVO)

    - (void)NN_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context {
        //1. 创建子类
        NSString *originClassName = NSStringFromClass(self.class);
        NSString *newClassName = [@"NSKVONotifying_" stringByAppendingString:originClassName];
        
        Class newClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
        objc_registerClassPair(newClass);//创建子类后,需要register
        
        //2. 创建方法,需要重写父类的setter方法
        class_addMethod(newClass, @selector(setName:), (IMP)(NN_SetMethod), "");
        
        //3. 更改类型
        object_setClass(self, newClass);
        
        [[self getObservers] setObject:observer forKey:keyPath];
    }
    
    // setter方法
    void NN_SetMethod(id self, SEL method, id setValue) {
        Class subClass = object_getClass(self); // 获取当前类
        Class superClass = class_getSuperclass(subClass); // 获取父类
        
        object_setClass(self, superClass); // 更改当前对象的类型
        ((void (*) (id, SEL, id)) objc_msgSend) (self, method, setValue); // 调用setter方法,进行数据更改
        object_setClass(self, subClass); // 更改回原来的子类
        
        // 触发回调observe更新了属性
        for (id objc in [self getObservers].allValues) {
            ((void (*) (id, SEL, id, id, id ,id)) objc_msgSend)(objc, @selector(observeValueForKeyPath:ofObject:change:context:), @"name", self, nil, nil);
    }
    
    - (NSMutableDictionary *)getObservers {
        NSMutableDictionary *observers = objc_getAssociatedObject(self, _cmd);
        if (!observers) {
            observers = [NSMutableDictionary dictionary];
            objc_setAssociatedObject(self, _cmd, observers, OBJC_ASSOCIATION_RETAIN);
        }
        return observers;
    }
}

@end
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 上半年有段时间做了一个项目,项目中聊天界面用到了音频播放,涉及到进度条,当时做android时候处理的不太好,由于...
    DaZenD阅读 3,052评论 0 26
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,856评论 0 9
  • 一、概述 KVO,即:Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则其观察...
    DeerRun阅读 10,152评论 11 33
  • [深入浅出Cocoa]详解键值观察(KVO)及其实现机理罗朝辉 (http://www.cppblog.com/k...
    Crazy2015阅读 714评论 0 1
  • 两天的学习很快就结束了,虽然时间短暂,但老师安排的课程紧张有序,真可谓紧锣密鼓,而又不乏趣味盎然。 首...
    等待飞翔的蜗牛阅读 1,129评论 0 2