前言本质
Automatic key-value observing is implemented using a technique called isa-swizzling
这计划的意思就是
自动的键值观察的实现基于 isa-swizzling
原理
1.KVO是基于runtime机制实现的
2.当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,
在这个派生类中重写基类中任何被观察属性的setter 方法。
派生类在被重写的setter方法内实现真正的通知机制
3.如果原类为Dog,那么生成的派生类名为NSKVONotifying_Dog
4.每个类对象中都有一个isa指针指向当前类,当一个类对象的第一次被观察,那么系统会偷偷将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter方法
5.键值观察通知依赖于NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey:一定会被调用,这就 会记录旧的值。而当改变发生后,didChangeValueForKey:会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。
一 KVO 的使用
#import "ViewController.h"
#import "Person.h"
@interface ViewController ()
@property (nonatomic,strong)Person *p1;
@property (nonatomic,strong)Person *p2;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
// 监听person 类的属性值
self.p1 = [[Person alloc]init];
self.p1.name = @"p1_dog";
self.p2 = [[Person alloc]init];
self.p2.name = @"p2_dog";
NSKeyValueObservingOptions val = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.p1 addObserver:self forKeyPath:@"name" options:val context:nil];
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
self.p1.name = @"p1_cate";
self.p2.name = @"p2_cate";
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
NSLog(@"%@ value changed from %@",keyPath,change);
}
@end
二 如何实现监听 ?
本质原因就是这两个对象的isa 指向不一样吗,主要是使用runtime 动态生成了一个派生类
1 如果未使用kvo 监听对象的时候
使用KVO 监听的话
三 代码验证
利用runtime 进行验证一下子,进入断电模式打印一下
self.p1 = [[Person alloc]init];
self.p1.name = @"p1_dog";
self.p2 = [[Person alloc]init];
self.p2.name = @"p2_dog";
// 打印添加监听之前 setName的方法
NSLog(@"监听之前%p--%p",[self.p1 methodForSelector:@selector(setName:)],[self.p1 methodForSelector:@selector(setName:)]);
NSKeyValueObservingOptions val = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.p1 addObserver:self forKeyPath:@"name" options:val context:nil];
NSLog(@"监听之后%p--%p",[self.p1 methodForSelector:@selector(setName:)],[self.p1 methodForSelector:@selector(setName:)]);
NSLog(@"end");
添加之后,setName 的实现走的是_NSSetObjectValueAndNotify 方法
四 KVO的 调用顺序
KVO
的调用顺序是:
- 调用
willChangeValueForKey:
- 调用原来的
setter
实现 - 调用
didChangeValueForKey:
也就是说 didChangeValueForKey:
内部必然是调用了 observer
的observeValueForKeyPath:ofObject:change:context:
方法。