欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

iOS:底层原理之 KVO & KVC

程序员文章站 2024-03-24 13:59:40
...

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];
}

iOS:底层原理之 KVO & KVC
[实例对象 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:
    iOS:底层原理之 KVO & KVC
    • 如通过 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

iOS:底层原理之 KVO & KVC

  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:

iOS:底层原理之 KVO & KVC

  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。
相关标签: iOS