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:。
- [super didChangeValueForKey]
问:如何手动出发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
- setValue:forKey:
- setKey —> _setKey
- 未找到 —> accessInstanceVariablesDirectly(是否允许直接赋值),return (BOOL)。
- return NO:调用 setValue:forUnderfinedKey:并抛出异常 NSUnknownKeyException。
- return YES:按照_key、_isKey、key、isKey 顺序进行查找。
- 如找到了,直接赋值。
- 如未找到成员变量:调用 setValue:forUnderfinedKey: 并抛出异常 NSUnknownKeyException。
KVC 会触发 KVO 监听,KVC setValue:forKey 方法调用 Setter 方法时如没有找到相应的 Setter 方法也可能触发 KVO。
原因:
- 触发了willChangeValueForKey
- self.stu->_age = 10;
- 触发了didChangeValueForKey
教程视频中这样解释:是否允许直接赋值,理解起来比较形象吧。
accessInstanceVariablesDirectly 方法的默认返回值是:YES
valueForKey:
- valueForKey:
- 按照 getKey->key->isKey->_key 顺序查找方法。
- 没有找到 查看 accessInstanceVariablesDirectly方法的返回值。
- return NO:调用 setValue:forUnderfinedKey:并抛出异常NSUnknownKeyException。
- return YES:按照 _key、_isKey、key、isKey顺序查找成员变量。
- 如找到了,直接赋值。
- 如未找到成员变量:调用 setValue:forUnderfinedKey: 并抛出异常 NSUnknownKeyException。
上一篇: NSNotificationCenter 通知使用方法详解
下一篇: 利用runtime实现KVO