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

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吧。

完结.

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容

  • 本文结构如下: Why? (为什么要用KVO) What? (KVO是什么) How? ( KVO怎么用) Mo...
    等开会阅读 1,641评论 1 21
  • 本文由我们团队的 纠结伦 童鞋撰写。 文章结构如下: Why? (为什么要用KVO) What? (KVO是什么...
    知识小集阅读 7,411评论 7 105
  • 上半年有段时间做了一个项目,项目中聊天界面用到了音频播放,涉及到进度条,当时做android时候处理的不太好,由于...
    DaZenD阅读 3,015评论 0 26
  • 本篇会对KVO的实现进行探究,不涉及太多KVO的使用方法,但是会有一些使用时的思考。 一、使用上的疑问 1.key...
    奋拓达阅读 504评论 0 2
  • 我和她的恋爱中,总是伴随着吵架、脾气。记的第一次吵架,也算不上吵架吧。只是两个人不欢而散,她先是不说话蹲...
    帅一阅读 418评论 1 0