KVO是key-value observing的缩写
KVO是Objective-C对观察者设计模式的实现
Apple使用isa混写(isa-swizzling)来实现KVO
通过isa混写就是,当我们注册一个观察者,调用- (void)addObserver:(NSObject*)observer forKeyPath:(NSString*)keyPath options:(NSKeyValueObservingOptions)options context:(nullablevoid*)context,去观察一个对象的属性,系统会在运行时动态为我们创建一个NSKVONotifying_A类,以前类的isa会指向新创建的NSKVONotifying_A类。NSKVONotifying_A就是原来类的子类,他会重写Setter方法,重写Setter方法会通知所有的观察者
下面我们代码实现一下
先创建两个类KObject和KObserver,通过名字我们可以看出两个类的作用
KObject.h
#import <Foundation/Foundation.h>
@interfaceKObject :NSObject
/** 注释 */
@property (nonatomic,assign) int value;
- (void)add;
@end
KObject.m
#import "KObject.h"
@implementation KObject
- (id)init{
self= [superinit];
if(self) {
_value=0;
}
return self;
}
- (void)add{
_value+=1;
}
@end
KObserver.m
#import "KObserver.h"
#import "KObject.h"
@implementation KObserver
- (void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context{
if([objectisKindOfClass:[KObjectclass]] && [keyPathisEqualToString:@"value"]) {
NSNumber *number = [change valueForKey:NSKeyValueChangeNewKey];
NSLog(@"value is %@",number);
}
}
引用两个类
KObject*obj = [[KObjectalloc]init];
KObserver*observer = [[KObserveralloc]init];
NSLog(@"obj_old : %s ",object_getClassName(obj));
[objaddObserver:observer forKeyPath:@"value" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"obj_new : %s",object_getClassName(obj));
obj.value=1;
打印结果
2018-11-14 17:02:20.890465+0800 CategrayTest[70728:13350625] obj_old : KObject
2018-11-14 17:02:20.890982+0800 CategrayTest[70728:13350625] obj_new : NSKVONotifying_KObject
2018-11-14 17:02:20.891274+0800 CategrayTest[70728:13350625] value is 1
我们可以看到未添加观察之前我们的obj属于KObject类,添加完Observer,属于NSKVONotifying_KObject类,这就是我们上面说的isa混写
我们刚才说系统为我们重写了Setter方法,具体内部怎么实现的的呢,我们看一下代码
- (void)setValue:(int)obj{
[self willChangeValueForKey:@"keyPath"];
[super setValue:obj];
[self didChangeValueForKey:@"keyPath"];
}
从代码中我们可以看到,系统为我们重写Setter方法,内部调用了willChangeValueForKey和didChangeValueForKey,其实调用didChangeValueForKey会触发添加观察者类,去调用observeValueForKeyPath。。。这个方法,来达到我们观察一个对象的属性的目的
还有一个问题,我们怎么给value赋值,才能出发Setter方法,通过点可以触发,那么通过kvc可以吗,我们试一下
obj.value=1;
[obj setValue:@2 forKey:@"value"];
打印结果
2018-11-14 17:24:36.286900+0800 CategrayTest[71027:13409991] value is 1
2018-11-14 17:24:36.287082+0800 CategrayTest[71027:13409991] value is 2
看来是可以的
如果直接赋值呢
obj.value=1;
[obj setValue:@2 forKey:@"value"];
[obj add];
打印结果
2018-11-14 17:28:42.525141+0800 CategrayTest[71091:13421105] value is 1
2018-11-14 17:28:42.525303+0800 CategrayTest[71091:13421105] value is 2
看来不行,add方法实现,上面代码中有
如果我们在属性赋值之前和之后分别加上willChangeValueForKey和didChangeValueForKey呢
add方法改装
- (void)add{
[self willChangeValueForKey:@"value"];
_value+=1;
[self didChangeValueForKey:@"value"];
}
obj.value=1;
[objsetValue:@2 forKey:@"value"];
[objadd];
打印结果
2018-11-14 17:31:11.251041+0800 CategrayTest[71147:13428256] value is 1
2018-11-14 17:31:11.251222+0800 CategrayTest[71147:13428256] value is 2
2018-11-14 17:31:11.251510+0800 CategrayTest[71147:13428256] value is 3
看来我们可以手动实现kvo
网友评论