如何优雅地使用 KVO

KVO 作为 iOS 中一种强大并且有效的机制,为 iOS 开发者们提供了很多的便利;我们可以使用 KVO 来检测对象属性的变化、快速做出响应,这能够为我们在开发强交互、响应式应用以及实现视图和模型的双向绑定时提供大量的帮助。

但是在大多数情况下,除非遇到不用 KVO 无法解决的问题,笔者都会尽量避免它的使用,这并不是因为 KVO 有性能问题或者使用场景不多,主要的原因是 KVO 的使用实在是太麻烦了。

使用 KVO 时,既需要进行注册成为某个对象属性的观察者,还要在合适的时间点将自己移除,再加上需要覆写一个又臭又长的方法,并在方法里判断这次是不是自己要观测的属性发生了变化,每次想用 KVO 解决一些问题的时候,作者的第一反应就是头疼,这篇文章会为各位为 KVO 所苦的开发者提供一种更优雅的解决方案。

使用 KVO

不过在介绍如何优雅地使用 KVO 之前,我们先来回忆一下,在通常情况下,我们是如何使用 KVO 进行键值观测的。

首先,我们有一个Fizz类,其中包含一个number属性,它在初始化时会自动被赋值为@0:

// Fizz.h@interfaceFizz:NSObject

@property(nonatomic,strong)NSNumber*number;

@end

// Fizz.m

@implementationFizz

-(instancetype)init{

if(self=[superinit]){

_number=@0;

}returnself;

}

@end

我们想在Fizz对象中的number对象发生改变时获得通知得到的和的值,这时我们就要祭出-addObserver:forKeyPath:options:context方法来监控number属性的变化:

Fizz*fizz=[[Fizz alloc]init];

[fizz addObserver:selfforKeyPath:@"number"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld          context:nil];

fizz.number=@2;

在将当前对象self注册成为fizz的观察者之后,我们需要在当前对象中覆写-observeValueForKeyPath:ofObject:change:context:方法:

-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary*)change context:(void*)context{

if([keyPath isEqualToString:@"number"]){

NSLog(@"%@",change);

}

}

在大多数情况下我们只需要对比keyPath的值,就可以知道我们到底监控的是哪个对象,但是在更复杂的业务场景下,使用context上下文以及其它辅助手段才能够帮助我们更加精准地确定被观测的对象。

但是当上述代码运行时,虽然可以成功打印出change字典,但是却会发生崩溃,你会在控制台中看到下面的内容:

2017-02-2623:44:19.666KVOTest[15888:513229]{

kind=1;

new=2;

old=0;

}

2017-02-2623:44:19.720KVOTest[15888:513229]***Terminating app due to uncaught exception'NSInternalInconsistencyException',reason:'An instance0x60800001dd20of class Fizz was deallocatedwhilekey value observers were still registered with it.Current observation info:(Context:0x0,Property:0x608000057400>)'

这是因为fizz对象没有被其它对象引用,在脱离viewDidLoad作用于之后就被回收了,然而在-dealloc时,并没有移除观察者,所以会造成崩溃。

我们可以使用下面的代码来验证上面的结论是否正确:

// Fizz.h

@interfaceFizz:NSObject

@property(nonatomic,strong)NSNumber*number;

@property(nonatomic,weak)NSObject*observer;

@end

// Fizz.m@implementationFizz

-(instancetype)init{

if(self=[superinit]){

_number=@0;

}

returnself;

}

-(void)dealloc{

[selfremoveObserver:self.observer forKeyPath:@"number"];

}

@end

在Fizz类的接口中添加一个observer弱引用来持有对象的观察者,并在对象-dealloc时将它移除,重新运行这段代码,就不会发生崩溃了。


由于没有移除观察者导致崩溃使用 KVO 时经常会遇到的问题之一,解决办法其实有很多,我们在这里简单介绍一个,使用当前对象持有被观测的对象,并在当前对象-dealloc时,移除观察者:

-(void)viewDidLoad{

[superviewDidLoad];

self.fizz=[[Fizz alloc]init];

[self.fizz addObserver:selfforKeyPath:@"number"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld                  context:nil];

self.fizz.number=@2;

}

-(void)dealloc{

[self.fizz removeObserver:selfforKeyPath:@"number"];

}

这也是我们经常使用来避免崩溃的办法,但是在笔者看来也是非常的不优雅,除了上述的崩溃问题,使用 KVO 的过程也非常的别扭和痛苦:

需要手动移除观察者,且移除观察者的时机必须合适

注册观察者的代码和事件发生处的代码上下文不同,传递上下文是通过void *指针;

需要覆写-observeValueForKeyPath:ofObject:change:context:方法,比较麻烦;

在复杂的业务逻辑中,准确判断被观察者相对比较麻烦,有多个被观测的对象和属性时,需要在方法中写大量的if进行判断;

虽然上述几个问题并不影响 KVO 的使用,不过这也足够成为笔者尽量不使用 KVO 的理由了。

优雅地使用 KVO

如何优雅地解决上一节提出的几个问题呢?我们在这里只需要使用 Facebook 开源的KVOController框架就可以优雅地解决这些问题了。

如果想要实现同样的业务需求,当使用 KVOController 解决上述问题时,只需要以下代码就可以达到与上一节中完全相同的效果:

[self.KVOController observe:self.fizz                    keyPath:@"number"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld                      block:^(id  _Nullable observer,id  _Nonnull object,NSDictionary*_Nonnull change){NSLog(@"%@",change);}];

我们可以在任意对象上获得KVOController对象,然后调用它的实例方法-observer:keyPath:options:block:就可以检测某个对象对应的属性了,该方法传入的参数还是非常容易理解的,在 block 中也可以获得所有与 KVO 有关的参数。

使用 KVOController 进行键值观测可以说完美地解决了在使用原生 KVO 时遇到的各种问题。

不需要手动移除观察者;

实现 KVO 与事件发生处的代码上下文相同,不需要跨方法传参数;

使用 block 来替代方法能够减少使用的复杂度,提升使用 KVO 的体验;

每一个keyPath会对应一个属性,不需要在 block 中使用if判断keyPath;

KVOController 的实现

KVOController 其实是对 Cocoa 中 KVO 的封装,它的实现其实也很简单,整个框架中只有两个实现文件,先来简要看一下 KVOController 如何为所有的NSObject对象都提供-KVOController属性的吧。

分类和 KVOController 的初始化

KVOController 不止为 Cocoa Touch 中所有的对象提供了-KVOController属性还提供了另一个KVOControllerNonRetaining属性,实现方法就是分类和 ObjC Runtime。

@interfaceNSObject(FBKVOController)

@property(nonatomic,strong)FBKVOController*KVOController;

@property(nonatomic,strong)FBKVOController*KVOControllerNonRetaining;

@end

从名字可以看出KVOControllerNonRetaining在使用时并不会持有被观察的对象,与它相比KVOController就会持有该对象了。

对于KVOController和KVOControllerNonRetaining属性来说,其实现都非常简单,对运行时非常熟悉的读者都应该知道使用关联对象就可以轻松实现这一需求。

-(FBKVOController*)KVOController{

id controller=objc_getAssociatedObject(self,NSObjectKVOControllerKey);if(nil==controller){controller=[FBKVOController controllerWithObserver:self];self.KVOController=controller;}returncontroller;}-(void)setKVOController:(FBKVOController*)KVOController{objc_setAssociatedObject(self,NSObjectKVOControllerKey,KVOController,OBJC_ASSOCIATION_RETAIN_NONATOMIC);}-(FBKVOController*)KVOControllerNonRetaining{id controller=objc_getAssociatedObject(self,NSObjectKVOControllerNonRetainingKey);

if(nil==controller){

controller=[[FBKVOController alloc]initWithObserver:selfretainObserved:NO];

self.KVOControllerNonRetaining=controller;

}

return controller;

}

-(void)setKVOControllerNonRetaining:(FBKVOController*)KVOControllerNonRetaining{

objc_setAssociatedObject(self,NSObjectKVOControllerNonRetainingKey,KVOControllerNonRetaining,OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

两者的setter方法都只是使用objc_setAssociatedObject按照键值简单地存一下,而getter中不同的其实也就是对于FBKVOController的初始化了。

到这里这个整个 FBKVOController 框架中的两个实现文件中的一个就介绍完了,接下来要看一下其中的另一个文件中的类KVOController。

KVOController 的初始化

KVOController是整个框架中提供 KVO 接口的类,作为 KVO 的管理者,其必须持有当前对象所有与 KVO 有关的信息,而在KVOController中,用于存储这个信息的数据结构就是NSMapTable。

为了使KVOController达到线程安全,它还必须持有一把pthread_mutex_t锁,用于在操作_objectInfosMap时使用。

再回到上一节提到的初始化问题,NSObject的属性FBKVOController和KVOControllerNonRetaining的区别在于前者会持有观察者,使其引用计数加一。

-(instancetype)initWithObserver:(nullable id)observer retainObserved:(BOOL)retainObserved{

self=[superinit];

if(nil!=self){

_observer=observer;NSPointerFunctionsOptions keyOptions=retainObserved?NSPointerFunctionsStrongMemory|NSPointerFunctionsObjectPointerPersonality:NSPointerFunctionsWeakMemory|NSPointerFunctionsObjectPointerPersonality;

_objectInfosMap=[[NSMapTable alloc]initWithKeyOptions:keyOptions valueOptions:NSPointerFunctionsStrongMemory|NSPointerFunctionsObjectPersonality capacity:0];

pthread_mutex_init(&_lock,NULL);}returnself;

}

在初始化方法中使用各自的方法对KVOController对象持有的所有实例变量进行初始化,KVOController和KVOControllerNonRetaining的区别就体现在生成的NSMapTable实例时传入的是NSPointerFunctionsStrongMemory还是NSPointerFunctionsWeakMemory选项。

KVO 的过程

使用KVOController实现键值观测时,大都会调用实例方法-observe:keyPath:options:block来注册成为某个对象的观察者,监控属性的变化:

-(void)observe:(nullable id)object keyPath:(NSString*)keyPath options:(NSKeyValueObservingOptions)options block:(FBKVONotificationBlock)block{

_FBKVOInfo*info=[[_FBKVOInfo alloc]initWithController:selfkeyPath:keyPath options:options block:block];

[self_observe:object info:info];

}

数据结构 _FBKVOInfo

这个方法中就涉及到另外一个私有的数据结构_FBKVOInfo,这个类中包含着所有与 KVO 有关的信息:

_FBKVOInfo在KVOController中充当的作用仅仅是一个数据结构,我们主要用它来存储整个 KVO 过程中所需要的全部信息,其内部没有任何值得一看的代码,需要注意的是,_FBKVOInfo覆写了-isEqual:方法用于对象之间的判等以及方便NSMapTable的存储。

如果再有点别的什么特别作用的就是,其中的state表示当前的 KVO 状态,不过在本文中不会具体介绍。

typedefNS_ENUM(uint8_t,_FBKVOInfoState){

_FBKVOInfoStateInitial=0,

_FBKVOInfoStateObserving,

_FBKVOInfoStateNotObserving,

};

observe 的过程

在使用-observer:keyPath:options:block:监听某一个对象属性的变化时,该过程的核心调用栈其实还是比较简单:

我们从栈底开始简单分析一下整个封装 KVO 的过程,其中栈底的方法,也就是我们上面提到的-observer:keyPath:options:block:初始化了一个名为_FBKVOInfo的对象:

-(void)observe:(nullable id)object keyPath:(NSString*)keyPath options:(NSKeyValueObservingOptions)options block:(FBKVONotificationBlock)block{

_FBKVOInfo*info=[[_FBKVOInfo alloc]initWithController:selfkeyPath:keyPath options:options block:block]

;[self_observe:object info:info];

}

在创建了_FBKVOInfo之后执行了另一个私有方法-_observe:info::


-(void)_observe:(id)object info:(_FBKVOInfo*)info{

pthread_mutex_lock(&_lock);

NSMutableSet*infos=[_objectInfosMap objectForKey:object];

_FBKVOInfo*existingInfo=[infos member:info];

if(nil!=existingInfo){

pthread_mutex_unlock(&_lock);

return;

}

if(nil==infos){

infos=[NSMutableSet set];

[_objectInfosMap setObject:infos forKey:object];}[infos addObject:info];

pthread_mutex_unlock(&_lock);

[[_FBKVOSharedController sharedController]observe:object info:info];

}

这个私有方法通过自身持有的_objectInfosMap来判断当前对象、属性以及各种上下文是否已经注册在表中存在了,在这个_objectInfosMap中保存着对象以及与对象有关的_FBKVOInfo集合:

在操作了当前KVOController持有的_objectInfosMap之后,才会执行私有的_FBKVOSharedController类的实例方法-observe:info::

-(void)observe:(id)object info:(nullable _FBKVOInfo*)info{

pthread_mutex_lock(&_mutex);

[_infos addObject:info];

pthread_mutex_unlock(&_mutex);

[object addObserver:selfforKeyPath:info->_keyPath options:info->_options context:(void*)info];

if(info->_state==_FBKVOInfoStateInitial){

info->_state=_FBKVOInfoStateObserving;

}elseif(info->_state==_FBKVOInfoStateNotObserving){

[object removeObserver:selfforKeyPath:info->_keyPath context:(void*)info];

}

}

_FBKVOSharedController才是最终调用 Cocoa 中的-observe:forKeyPath:options:context:方法开始对属性的监听的地方;同时,在整个应用运行时,只会存在一个_FBKVOSharedController实例:

+(instancetype)sharedController{

static_FBKVOSharedController*_controller=nil;

staticdispatch_once_t onceToken;dispatch_once(&onceToken,^{_controller=[[_FBKVOSharedController alloc]init];});

return_controller;

}

这个唯一的_FBKVOSharedController实例会在 KVO 的回调方法中将事件分发给 KVO 的观察者。

-(void)observeValueForKeyPath:(nullable NSString*)keyPath                      ofObject:(nullable id)object                        change:(nullable NSDictionary*)change                      context:(nullablevoid*)context{

_FBKVOInfo*info;pthread_mutex_lock(&_mutex);

info=[_infos member:(__bridge id)context];pthread_mutex_unlock(&_mutex);

FBKVOController*controller=info->_controller;id observer=controller.observer;

if(info->_block){

NSDictionary*changeWithKeyPath=change;

if(keyPath){

NSMutableDictionary*mChange=[NSMutableDictionary dictionaryWithObject:keyPath forKey:FBKVONotificationKeyPathKey];

[mChange addEntriesFromDictionary:change];

changeWithKeyPath=[mChange copy];

}

info->_block(observer,object,changeWithKeyPath);

}elseif(info->_action){

[observer performSelector:info->_action withObject:change withObject:object];

}else{

[observer observeValueForKeyPath:keyPath ofObject:object change:change context:info->_context];

}

}

在这个-observeValueForKeyPath:ofObject:change:context:回调方法中,_FBKVOSharedController会根据 KVO 的信息_KVOInfo选择不同的方式分发事件,如果观察者没有传入 block 或者选择子,就会调用观察者 KVO 回调方法。

上图就是在使用 KVOController 时,如果一个 KVO 事件触发之后,整个框架是如何对这个事件进行处理以及回调的。

如何 removeObserver

在使用 KVOController 时,我们并不需要手动去处理 KVO 观察者的移除,因为所有的 KVO 事件都由私有的_KVOSharedController来处理;

当每一个KVOController对象被释放时,都会将它自己持有的所有 KVO 的观察者交由_KVOSharedController的-unobserve:infos:方法处理:

-(void)unobserve:(id)object infos:(nullable NSSet<_FBKVOInfo*>*)infos{

pthread_mutex_lock(&_mutex);

for(_FBKVOInfo*infoininfos){

[_infos removeObject:info];

}

pthread_mutex_unlock(&_mutex);

for(_FBKVOInfo*infoininfos){

if(info->_state==_FBKVOInfoStateObserving){

[object removeObserver:selfforKeyPath:info->_keyPath context:(void*)info];

}

info->_state=_FBKVOInfoStateNotObserving;

}

}

该方法会遍历所有传入的_FBKVOInfo,从其中取出keyPath并将_KVOSharedController移除观察者。

除了在KVOController析构时会自动移除观察者,我们也可以通过它的实例方法-unobserve:keyPath:操作达到相同的效果;不过在调用这个方法时,我们能够得到一个不同的调用栈:

功能的实现过程其实都是类似的,都是通过-removeObserver:forKeyPath:context:方法移除观察者:

-(void)unobserve:(id)object info:(nullable _FBKVOInfo*)info{

pthread_mutex_lock(&_mutex);

[_infos removeObject:info];

pthread_mutex_unlock(&_mutex);

if(info->_state==_FBKVOInfoStateObserving){

[object removeObserver:selfforKeyPath:info->_keyPath context:(void*)info];

}

info->_state=_FBKVOInfoStateNotObserving;

}

不过由于这个方法的参数并不是一个数组,所以并不需要使用for循环,而是只需要将该_FBKVOInfo对应的 KVO 事件移除就可以了。

总结

KVOController 对于 Cocoa 中 KVO 的封装非常的简洁和优秀,我们只需要调用一个方法就可以完成一个对象的键值观测,同时不需要处理移除观察者等问题,能够降低我们出错的可能性。

在笔者看来 KVOController 中唯一不是很优雅的地方就是,需要写出object.KVOController才可以执行 KVO,如果能将KVOController换成更短的形式可能看起来更舒服一些:

[self.kvo observer:keyPath:options:block:];

不过这并不是一个比较大的问题,同时也只是笔者自己的看法,况且不影响 KVOController 的使用,所以各位读者也无须太过介意。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容

  • KVO 作为 iOS 中一种强大并且有效的机制,为 iOS 开发者们提供了很多的便利;我们可以使用 KVO 来检测...
    Draveness阅读 6,896评论 11 59
  • https://draveness.me/kvocontroller
    扛支枪阅读 143评论 0 0
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,709评论 0 9
  • 猜想runloop内部是如何实现的?一般来讲,一个线程一次只能执行一个任务,执行完成后线程就会退出。如果我们需要一...
    笔笔请求阅读 421评论 0 0
  • 上半年有段时间做了一个项目,项目中聊天界面用到了音频播放,涉及到进度条,当时做android时候处理的不太好,由于...
    DaZenD阅读 3,017评论 0 26