美文网首页iOS干货iOS开发程序员
iOS - 手把手带你一步一步实现KVO

iOS - 手把手带你一步一步实现KVO

作者: RaInVis | 来源:发表于2017-07-04 14:17 被阅读247次
    MacDown Screenshot

    前言

    KVO

    即:Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者。

    用法&原理

    MacDown Screenshot

    KVO实现原理很多博客都有很详细的介绍,这里我就不再重复的阐述了,推荐几篇博文供大家学习.

    来自sunnyxx大神的博客: objc kvo简单探索

    来自objc中国的文章: KVC 和 KVO

    正序

    实现KVO的思路

    在了解完KVO的原理过后,通过分析得出,大体上的实现思路是这样的:

    • 当注册一个观察者时,首先要动态创建被监听对象的类的一个派生类(子类)
    • 将原类的实例变量的isa指针指向新创建的派生类
    • 为了"混淆视听",伪装生成了派生类,重写class方法,使其返回派生类的父类(也就是被监听对象的类)
    • 重写派生类的setter方法,在里面通知被监听的值的改变

    具体核心代码实现(部分)

    添加观察者

        // 获取类&类名
        Class class = object_getClass(self);
        NSString *className = NSStringFromClass(class);
        
        // 判断是否已经生成KVO的派生类(前缀判断)
        if (![className hasPrefix:TFKVOClassPrefix]) {
            // 生成派生类
            class = [self creatNewClassWithInitialClass:className];
            // 设置对象的类为生成的派生类
            object_setClass(self, class);
        }
        // 判断是否已经实现重写了set方法
        if (![self hasSelector:setterSelector]) {
            const char *types = method_getTypeEncoding(setterMethod);
            // 重写set方法添加监听
            class_addMethod(class, setterSelector, (IMP)kvo_setter, types);
        }
         // 动态给注册者绑定数组,数组里面包含KVO信息(观察的observer,key,block)
        TFObservationInfo *info = [[TFObservationInfo alloc] initWithObserver:observer Key:key block:block];
        NSMutableArray *observers = objc_getAssociatedObject(self, (__bridge const void *)(TFKVOAssociatedObserverKey));
        if (!observers) {
            observers = [NSMutableArray array];
            objc_setAssociatedObject(self, (__bridge const void *)(TFKVOAssociatedObserverKey), observers, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
        }
        [observers addObject:info];
    
    

    动态创建派生类具体实现方法

        Class cls = NSClassFromString(KvoClassName);
        if (cls) { // 如果已经存在新创建的派生类,直接返回
            return cls;
        }
        Class initialClass = object_getClass(self);
        // 动态创建类
        Class kvoClass = objc_allocateClassPair(initialClass, KvoClassName.UTF8String, 0);
        // 得到类的实例方法
        Method classMethod = class_getInstanceMethod(kvoClass, @selector(class));
        // 获取方法的Type字符串(包含参数类型和返回值类型)
        const char *types = method_getTypeEncoding(classMethod);
        // 重写class方法
        class_addMethod(kvoClass, @selector(class), (IMP)kvo_class, types);
        // 注册创建的类
        objc_registerClassPair(kvoClass);
        
    
    

    重写setter方法

        // 构建 objc_super 的结构体
        struct objc_super superclass = {
            .receiver = self,
            .super_class = class_getSuperclass(object_getClass(self))
        };
        
        // 向父类发送set消息
        void (*objc_msgSendSuperCasted)(void *, SEL, id) = (void *)objc_msgSendSuper;
        objc_msgSendSuperCasted(&superclass, _cmd, newValue);
        // 调用完后,获取绑定的info,调用block回调
        NSMutableArray *observers = objc_getAssociatedObject(self, (__bridge const void *)(TFKVOAssociatedObserverKey));
        for (TFObservationInfo *info in observers) {
            if ([info.key isEqualToString:getterName]) {
                dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
                    info.block(self, getterName, oldValue, newValue);
                });
            }
        }
    
    

    移除观察者

      NSMutableArray *observers = objc_getAssociatedObject(self, (__bridge const void *)(TFKVOAssociatedObserverKey));
        TFObservationInfo *removeInfo;
        for (TFObservationInfo* info in observers) {
            if (info.observer == observer && [info.key isEqual:key]) {
                removeInfo = info;
                break;
            }
        }
        [observers removeObject:removeInfo];
    
    

    完整实现源码&详细注释&测试DEMO

    前往github下载源码

    尾言

    就笔者而言,在实际开发中,很少使用KVO提供的API,虽然他很强大,但是还有一些不足.比如,你只能通过重写 -observeValueForKeyPath:ofObject:change:context:方法来获得通知,想要自定义方法或者使用block实现,都是不允许的.而且你还要处理父类的情况,例如父类同样监听同一个对象的同一个属性。虽然context 这个参数就是干这个的,也可以解决这个问题 - 在 -addObserver:forKeyPath:options:context:传进去一个父类不知道的context,但是也是很麻烦的一件事,不是吗?

    有不少人都觉得官方 KVO 不好使的。Mike Ash 的
    Key-Value Observing Done Right,以及获得不少分享讨论的KVO Considered Harmful都把KVO"批判"了一把。所以在实际开发中 KVO 使用的情景并不多,更多时候还是用 Delegate 或 NotificationCenter吧。

    完结.

    如文中有不对的地方,还请及时回复修正,谢谢观赏!

    相关文章

      网友评论

      本文标题:iOS - 手把手带你一步一步实现KVO

      本文链接:https://www.haomeiwen.com/subject/jkvthxtx.html