KVO的全称是Key-Value Observing,即"键值监听",可以用于监听某个对象属性值得改变。
F、题:iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么)
答:
- 当一个对象使用了KVO监听,iOS系统会修改这个对象的isa指针,改为指向一个全新的通过Runtime动态创建的子类(NSKVONotifying_xxx)。
- 子类拥有自己的set方法实现,set方法实现内部会顺序调用
willChangeValueForKey
方法、[super setAxx:]
、didChangeValueForKey
方法。 - 在
didChangeValueForKey
方法内部又会调用监听器的observeValueForKeyPath:ofObject:change:context:
方法 - 这几个方法合起来就是Foundation的C方法
_NSSetInt(Double)ValueAndNotify
方法。
拓展:
//给person对象添加KVO监听
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.person addObserver:self forKeyPath:@"age" options:options context:@"123"];
[self.person addObserver:self forKeyPath:@"height" options:options context:@"456"];
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
NSLog(@"监听到%@的%@属性值改变了 - %@---content-%@", object, keyPath, change, context);
}
- (void)dealloc {
[self.person removeObserver:self forKeyPath:@"age"];
[self.person removeObserver:self forKeyPath:@"height"];
}
2019-01-26 17:22:15.315762+0800 InterViewOC[63236:684490] 监听到<LQPerson: 0x600001d8af90>的age属性值改变了 - {
kind = 1;
new = 20;
old = 10;
}---content-123
2019-01-26 17:22:15.315972+0800 InterViewOC[63236:684490] 监听到<LQPerson: 0x600001d8af90>的height属性值改变了 - {
kind = 1;
new = 21;
old = 11;
}---content-456
题:如何手动触发KVO
答:被监听的属性的值被修改时,就会自动触发KVO。如果想要手动触发KVO,则需要我们手动调用willChangeValueForKey和didChaneValueForKey方法即可在不改变属性值的情况下手动触发KVO,并且这两个方法缺一不可。(应该是在didChaneValueForKey方法中判断过前面是否调用过willChangeValueForKey,所以两者缺一不可)
LQPerson *p1 = [[LQPerson alloc] init];
p1.age = 1.0;
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[p1 addObserver:self forKeyPath:@"age" options:options context:@"789"];
[p1 willChangeValueForKey:@"age"];
[p1 didChangeValueForKey:@"age"];
[p1 removeObserver:self forKeyPath:@"age"];
2019-01-26 17:22:15.315972+0800 InterViewOC[63236:684490] 监听到<LQPerson: 0x600001d8af90>的height属性值改变了 - {
kind = 1;
new = 1;
old = 1;
}---content-789
注意:不是每个属性都能监听的,这个取决于代码如何实现的。
题:直接修改成员变量会触发KVO么?
答:不会触发KVO,因为直接修改成员变量并没有调用setXxx方法。
除非使用[self willChangeValueForKey:@"xxx"]; 修改成员变量; [self didChangeValueForKey:@"xxx"]; 才会触发KVO。
_NSSet*ValueAndNotify的内部调用流程:
[self willChangeValueForKey:@"xxx"]; // 调用willChangeValueForKey [super setXxx:]; // 原来的setter实现 [self didChangeValueForKey:@"xxx"]; // 调用didChangeValueForKey
didChangeValueForKey
内部会调用observer
的ObserverValueForKeyPath:ofObject:change:context
- (void)willChangeValueForKey:(NSString *)key { NSLog(@"willChangeValueForKey --- begin"); [super willChangeValueForKey:key]; NSLog(@"willChangeValueForKey --- end"); } - (void)didChangeValueForKey:(NSString *)key { NSLog(@"didChangeValueForKey --- begin"); [super didChangeValueForKey:key]; NSLog(@"didChangeValueForKey --- end"); }
打印类对象中的方法列表:
- (void)printMethodWithClass:(Class)cls {
unsigned int count;
// 获得方法数组
Method *methods = class_copyMethodList(cls, &count);
// 存储方法名
NSMutableString *methodNames = [NSMutableString string];
// 遍历所有方法
for (int i = 0; i < count; i++) {
// 获得方法
Method method = methods[i];
// 获得方法名
NSString *mname = NSStringFromSelector(method_getName(method));
// 拼接方法名
[methodNames appendFormat:@"%@ ", mname];
}
// 释放内存
free(methods);
}