KVO

作者: 楠若寺 | 来源:发表于2018-05-16 23:17 被阅读0次

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

相关文章

  • iOS原理篇(一): KVO实现原理

    KVO实现原理 什么是 KVO KVO 基本使用 KVO 的本质 总结 一 、 什么是KVO KVO(Key-Va...

  • 04. KVO使用,原理,本质

    问题 KVO日常使用 KVO原理(KVO本质是什么) 如何手动触发KVO 直接修改成员变量会触发KVO吗 KVO图...

  • 20.iOS底层学习之KVO 原理

    本篇提纲1、KVO简介;2、KVO的使用;3、KVO的一些细节;4、KVO的底层原理; KVO简介 KVO全称Ke...

  • 深入理解KVO

    iOS | KVO | Objective-C KVO的本质是什么,如何手动触发KVO? 1.什么是KVO KVO...

  • OC语法:KVO的底层实现

    一、KVO是什么二、怎么使用KVO三、KVO的底层实现四、KVO常见面试题 一、KVO是什么 KVO全称Key-V...

  • KVO基本使用

    分三部分解释KVO一.KVO基本使用二.KVO原理解析三.自定义实现KVO 一、KVO基本使用 使用KVO,能够非...

  • KVO 解析

    KVO解析(一) —— 基本了解KVO解析(二) —— 一个简单的KVO实现KVO解析(三) —— KVO合规性K...

  • KVO

    目录 1. KVO的使用1.1 KVO基本使用方法1.2 KVO手动触发模式1.3 KVO属性依赖1.4 KVO容...

  • OC语言之KVO与KVC

    KVO 什么是KVO? KVO 是 Key-value observing(键值观察)的缩写。 KVO是Objec...

  • 可能碰到的iOS笔试面试题(7)--KVO-KVC

    KVC-KVO KVC的底层实现? KVO的底层实现? 什么是KVO和KVC? KVO的缺陷? KVO是一个对象能...

网友评论

      本文标题:KVO

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