KVO的实现原理

KVO(Key-Value Observing)通过Runtime动态生成子类来实现属性监听。当添加KVO后,对象的isa指针会指向一个新创建的子类,该子类重写了原类的setter方法。在setter内部,通过`_NSSetIntValueAndNotify`调用`willChangeValueForKey`和`didChangeValueForKey`来通知观察者。手动触发KVO需要调用这两个方法,而直接修改成员变量不会触发KVO。

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

KVO:key-value observing 键值监听

未添加KVO监听的类, person.isa = PJPerson
给PJPerson添加KVO监听之后,RunTime会动态的添加一个类NSKVONotifying_PJPerson,这个类是PJPerson的子类 person.isa = NSKVONotifying_PJPerson

KVO的本质其实就是:利用runtime的动态属性给PJPerson添加一个子类,person通过isa指针找到这个子类的setAge类实现,这个setAge方法里面做了一些特殊的处理

  1. 调用_NSSetIntValueAndNotify方法 (Foundation`_NSSetIntValueAndNotify) (通过[self.person1 methodForSelector:@selector(setAge:)] p (IMP)0x10816bcf2 就能获取到)
  2. NSSetIntValueAndNotify的实现
    2.1 [self willChangeValueForKey:@“age”];
    2.2 [super setAge:age];
    2.3 [self didChangeValueForKey:@“age”];
    2.3.1 didChangeValueForKey的实现 通知监听器[observe observeValueForKeyPath:key ofObject:self change:0 context:nil]
  3. 监听到属性的改变

-(void)setAge:(int)age {

// Foundation
_NSSetIntValueAndNotify();

}
object_getClass(self) 返回的才是真实的类

  • (Class)class {
    // 屏蔽内部实现, 隐藏了NSKVONotifying_PJPerson类的实现
    }

  • (void)dealloc {
    // 做一些善后工作
    }

  • (BOOL)_isKVOA {
    return YES;
    }

利用RunTime打印类内部所有的方法

  • (void)printMethodNamesOfClass:(Class)cls {

    unsigned int count;
    NSMutableString *methodNames = [NSMutableString string];
    Method *methodList = class_copyMethodList(cls, &count);
    for (int i=0; i<count; i++) {
    Method method = methodList[i];

      NSString *methodName = NSStringFromSelector(method_getName(method));
      
      [methodNames appendString:methodName];
      [methodNames appendString:@", "];
    

    }

    free(methodList);

    NSLog(@"%@, %@", cls, methodNames);
    }

KVO的本质是什么?
利用runtime的api动态的生成一个子类,并且让instance对象的isa指针指向它
当修改instance对象的属性时,会调用Foundation的`_NSSetIntValueAndNotify函数, 在_NSSetIntValueAndNotify函数内部
willChangeValueForKey:
调用父类原来的setter方法
didChangeValueForKey:
内部触发监听器(Oberser)的监听方法(observeValueForKeyPath:ofObject: change: context:)
KVO的本质其实就是改变了setAge方法的实现,能不能监听,其实就是看有没有setter方法
说到底就是修改了isa指针的指向,重写了setAge方法,替换了它的实现_NSSetIntValueAndNotify
重写class,dealloc方法

如何手动触发KVO
手动调用willChangeValueForKey和didChangeValueForKey

直接修改成员变量会触发KVO吗?
不会触发
切记,成员变量不是属性。
但可以手动的去触发它
[self.person1 willChangeValueForKey:@“age”];
self.person1->_age = 22;
[self.person1 didChangeValueForKey:@“age”];

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值