1、什么是KVO
KVO的全称是Key-Value Observing,俗称“键值监听”,可以用于监听某个对象属性值的改变
2、如何使用KVO
第一步:创建一个Person类 然后添加一个name属性
//.h文件
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface Person : NSObject
@property (nonatomic,strong)NSString *name;
@end
NS_ASSUME_NONNULL_END
//.m文件
#import "Person.h"
@implementation Person
@end
第二步 在控制器中初始化这个Person
#import "ViewController.h"
#import "Person.h"
@interface ViewController ()
@property (nonatomic,strong)Person *person1;
@property (nonatomic,strong)Person *person2;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.person1 = [[Person alloc] init];
self.person1.name = @"张三";
self.person2 = [[Person alloc] init];
self.person2.name = @"王五";
//给person1的name添加监听
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld;
[self.person1 addObserver:self forKeyPath:@"name" options:options context:@"这是名字的标记"];
/**
注意:context 其实就是一个标示符,如果一个对象需要监听多个属性值的时候,
可以使用context来区分是哪一个值发生了改变。
可以使用字符串来定义当前那个属性发生了改变,
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context;
方法中的context 就是从添加监听的时候传过来的。如添加监听的时候context 传的是:这是名字的标记
那么在触发函数中接收的context也是:这是名字的标记
*/
}
#pragma mark -- 当self.person.name 发生改变的时候就会触发这个方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"新值:%@--旧值:%@--标记:%@",change[@"new"],change[@"old"],context);
}
#pragma mark -- 点击屏幕改变name
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
self.person1.name = @"李四";
self.person2.name = @"赵六";
}
#pragma mark -- 当控制器销毁的时候直接移除监听器
- (void)dealloc{
[self.person1 removeObserver:self forKeyPath:@"name"];
}
@end
如上代码即完成了self.person1.name的监听
3、KVO底层实现
代码如2中的代码,创建两个person的实例变量 person1和person2,使用KVO对person1的name进行监听,
点击屏幕的时候 打印 person1和person2 的isa 结果如图所示
如上图所示:
person1的isa 指向的是 NSKVONotifying_Person
person2的isa 指向的是 Person
疑问:
实例对象的isa指向的类对象,怎么person1的isa指向的却是 NSKVONotifying_Person 这个是什么玩意?
区别:
person1 与 person2 唯一的区别就是 person1添加了KVO监听 而person2没有被监听。
结论:
1、当实例对象 进行KVO观察时候,会利用RuntimeAPI动态生成一个子类,然后将对象的isa指向新生成的子类
2、KVO本质上是监听属性的setter方法,只要被观察对象有成员变量和对应的set方法,
就会调用Foundation的_NSSetValueAndNotify函数这个函数内部会执行 willChangeVlaueForKey函数、
父类的setter方法 和didChangeVlaueForKey的方法
(didChangeVlaueForKey 方法内部会触发监听器的observeValueForKeyPath: ofObject: context:函数
3、子类会重写父类的set、class、dealloc、_isKVOA方法
4、当观察对象移除所有的监听后,会将观察对象的isa指向原来的类
5、当观察对象的监听全部移除后,动态生成的类不会注销,而是留在下次观察时候再使用,避免反复创建中间子类
person1 isa 指针指向如下图所示
person2 isa 指针指向如下图所示
4、如何手动出发KVO
由KVO的底层原理可以看出 触发KVO主要就是 runtime动态生成的子类重写了set方法 在set方法中
首先调用Foundation的_NSSetValueAndNotify函数这个函数内部会执行 willChangeVlaueForKey函数
父类的setter方法
didChangeVlaueForKey的方法
在didChangeVlaueForKey方法调用的时候就会触发监听器的
observeValueForKeyPath: ofObject: context:函数
所以想要手动触发KVO 就必须手动调用willChangeVlaueForKey 和didChangeVlaueForKey方法
5、直接修改成员变量会不会触发KVO
KVO 的本质就是重写属性的setter方法,如果直接修改成员变量就根本没有去触发setter方法
所以直接修改成员变量并不会触发KVO