iOS:底层原理之 KVO & KVC

KVO

通过 KVO 监听一个类的某键值属性,runtime 就会在运行时创建一个的子类 NSKVONotifying_(类名) ,如设置 Student 的 age 属性值,会调用子类中的 setAge 方法,而 setAge 方法实际调用了 Foundation 框架的 _NSSetIntValueAndNotify 方法。
Foundation 框架的 _NSSetIntValueAndNotify 方法的伪代码实现可以认为是:

void _NSSetIntValueAndNotify( )
{
    [self willChangeValueForKey:@"age"];
    [super setAge:age];
    [self didChangeValueForKey:@"age"];
}
- (void)didChangeValueForKey:(NSString *)key{
    [observe observeValueForKeyPath:key ofObject:self change:nil context:nil];
}

在这里插入图片描述
[实例对象 methodForSelector:(setAge:)]; 可以将其内存地址,在 LLDB 中通过【p (IMP)内存地址】可以打印出类方法的具体实现。

_NSSet*ValueAndNotify 的内部实现(*是指Int、double、float)

根据以下代码及 LLDB Log 可得出:

- (void)willChangeValueForKey:(NSString *)key
{
    [super willChangeValueForKey:key];
    NSLog(@"will%@",key);
}
- (void)didChangeValueForKey:(NSString *)key
{
    NSLog(@"didChangeValueForKey---begin:%@",key);
    [super didChangeValueForKey:key];     // observeValueForKeyPath 方法是在此方法中执行的。或说:监听器是在此方法里面调用的监听方法
    NSLog(@"didChangeValueForKey---end:%@",key);
}

LLDB Log:

2020-01-02 18:10:49.871108+0800 Test_2[34769:3912538] didChangeValueForKey---begin:age
2020-01-02 18:10:49.871349+0800 Test_2[34769:3912538] 监听<Student: 0x600002a6cee0> 的 age 发生改变----{
    kind = 1;
    new = 30;
    old = 10;
}---context:haha
2020-01-02 18:10:49.871456+0800 Test_2[34769:3912538] didChangeValueForKey---end:age

子类 NSKVONotifying_(类名) 中不仅仅重写了 setAge:方法,还重写了 class、dealloc、_isKVOA(返回BOOL) 方法

  • _isKVOA(返回BOOL) return YES;说明是使用了KVO
  • dealloc :做一些收尾的工作。
  • class:
    在这里插入图片描述
    • 如通过 object_getClass(执行 KVO 监听的实例对象) 获取到的类对象为子类 NSKVONotifying_(类名);
    • 通过调用 class 方法获取类对象,所获取到的类对象为(类名)
    • 目的是屏蔽内部实现。

NSObject 的 class 伪代码是:调用 object_getClass。

C 语言的数组如果是 creat 或 copy 出来的都需要进行 free 释放。

问:iOS用什么方式实现对一个对象的 KVO?(KVO的本质是什么?)
答:

  • 利用 Runtime API 的动态特性,动态创建一个子类 NSKVONotifying_(类名),并且让 Instance 对象的isa 指针指向这个子类。
  • 当修改 Instance 对象的属性时,会调用 Foundation 框架的 _NSSetXXXValueAndNotify函数。
  • _NSSetXXXValueAndNotify 函数的内部实现:
    • willChangeValueForKey
    • [super setAge:age]; // 调用父类的 setter 方法
    • didChangeValueForKey
      • [super didChangeValueForKey]
        • 内部触发监听器(Observe)的监听方法: - (void)observeValueForKeyPath:ofObject: change:context:。

问:如何手动出发KVO?(不修改属性值)
答:

  • 手动触发:willChangeValueForKey
  • 手动触发:didChangeValueForKey
  • 例:[self.stu1 willChangeValueForKey:@“age”];
  • 例:[self.stu1 didChangeValueForKey:@“age”];

问:直接修改成员变量是否会触发 KVO 监听?
答:
不会

Runtime 的动态生成类:objc_allocateClassPair() and before objc_registerClassPair()(此处印象有点模糊,后续再查)

KVC

  • setValue:forKey:
  • setValue:forKeyPath:
  • valueForKey:
  • valueForKeyPath:

[self.stu1 setValue:@10 forKey:@“number”];
[self.stu1 setValue:@30 forKeyPath:@“teacher.age”];
NSNumber *number = [self.stu1 valueForKey:@“number”];
NSNumber *teacherAge = [self.stu1 valueForKeyPath:@“teacher.age”];

SetValue

在这里插入图片描述

  1. setValue:forKey:
  2. setKey —> _setKey
  3. 未找到 —> accessInstanceVariablesDirectly(是否允许直接赋值),return (BOOL)。
  4. return NO:调用 setValue:forUnderfinedKey:并抛出异常 NSUnknownKeyException。
  5. return YES:按照_key、_isKey、key、isKey 顺序进行查找。
  6. 如找到了,直接赋值。
  7. 如未找到成员变量:调用 setValue:forUnderfinedKey: 并抛出异常 NSUnknownKeyException。

KVC 会触发 KVO 监听,KVC setValue:forKey 方法调用 Setter 方法时如没有找到相应的 Setter 方法也可能触发 KVO。
原因:

  • 触发了willChangeValueForKey
  • self.stu->_age = 10;
  • 触发了didChangeValueForKey

教程视频中这样解释:是否允许直接赋值,理解起来比较形象吧。
accessInstanceVariablesDirectly 方法的默认返回值是:YES

valueForKey:

在这里插入图片描述

  1. valueForKey:
  2. 按照 getKey->key->isKey->_key 顺序查找方法。
  3. 没有找到 查看 accessInstanceVariablesDirectly方法的返回值。
  4. return NO:调用 setValue:forUnderfinedKey:并抛出异常NSUnknownKeyException。
  5. return YES:按照 _key、_isKey、key、isKey顺序查找成员变量。
  6. 如找到了,直接赋值。
  7. 如未找到成员变量:调用 setValue:forUnderfinedKey: 并抛出异常 NSUnknownKeyException。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值