代码自己实现,深入探究KVO的内部实现

本文介绍了一种Objective-C中键值观察(KVO)机制的自定义实现方式。通过重写`addObserver:forKeyPath:options:context:`方法,并修改对象的类定义和isa指针来实现对属性变更的观察。此外,还展示了如何添加自定义的setter方法以通知观察者。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

#import "NSObject+KVO.h"
#import <objc/message.h>

@implementation NSObject_KVO


- (void)WK_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context{

    /*
     自定义子类
     重写方法
     修改isa指针
     */


    NSString *oldClassName =   NSStringFromClass([self class]);
    NSString *newClassName =  [ @"WKKVO_" stringByAppendingString:oldClassName];

    const char *newName = [newClassName UTF8String];

    Class Mycalss =  objc_allocateClassPair([self class], newName, 0);


    //添加set方法。相当于重写!!
    /*
     子类没有这个方法就会去父类找这个方法,但是并不代表子类拥有了这个方法。
     */

    class_addMethod(Mycalss, NSSelectorFromString(@"setName"), (IMP)setName, "v@:@");

    //注册这个类
    objc_registerClassPair(Mycalss);

    //修改self的isa指针
    object_setClass(self, Mycalss);

    //将观察者保存到当前对象
    objc_setAssociatedObject(self, (__bridge const void *)@"objc", observer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}


void  setName (id self , SEL _cmd ,NSString *newName){

    //保存当前类型
    id class =  [self class];
    //改变isa指针
    object_setClass(self, class_getSuperclass(class));
    //调用父类set方法
    objc_msgSend(self, @selector(setName:),newName);



    //拿出观察者
   id objc =  objc_getAssociatedObject(self, (__bridge const void *)@"objc");

   //通知观察者
    objc_msgSend(objc, @selector(observeValueForKeyPath:ofObject:change:context:), @"name",self,nil,nil);

    //改回子类类型
    object_setClass(self, class);



    /*
     将XCode升级到6后,报Too many arguments to function call, expected 0, have *,在XCode5.1里能编译通过的,到xcode6就报错

     objc_msgSend(objc, @selector(observeValueForKeyPath:ofObject:change:context:), @"name",self,nil,nil);

     Too many arguments to function call, expected 0, have *

     问了下度娘,

     选中项目 - Project - Build Settings - ENABLE_STRICT_OBJC_MSGSEND 将其设置为 NO 即可
     */


}


@end
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值