KVO和手动调用KVO

iOS 中的KVO分为自动监听和手动触发两种方式,自动监听很简单,不在此分析,这里主要说一下手动触发的实现,面试容易被问到

一、要点有以下几点:

1、重写监听属性的set、get方法

2、重写 +(BOOL)automaticallyNotifiesObserverForKey:(NSString *)key

3、在set方法中在赋值的前后分别调用:willChangeValueForKey和didChangeValueForKey

4、实现willChangeValueForKey和didChangeValueForKey方法


二、这里主要说下 +(BOOL)automaticallyNotifiesObserverForKey:(NSString *)key中的实现:



1、addObserver:forKeyPath:options:context:各个参数的作用分别是什么,observer中需要实现哪个方法才能获得KVO回调?

// 添加键值观察

/*1 观察者,负责处理监听事件的对象

2 观察的属性

3 观察的选项

4 上下文

*/

[self.personaddObserver:selfforKeyPath:@"name"options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOldcontext:@"Person Name"];

observer中需要实现一下方法:


// 所有的 kvo 监听到事件,都会调用此方法

/*1. 观察的属性

2. 观察的对象

3. change 属性变化字典(新/旧)

4. 上下文,与监听的时候传递的一致

*/

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


2、如何手动触发一个value的KVO

所谓的“手动触发”是区别于“自动触发”:

自动触发是指类似这种场景:在注册KVO之前设置一个初始值,注册之后,设置一个不一样的值,就可以触发了。

想知道如何手动触发,必须知道自动触发KVO的原理:

键值观察通知依赖于NSObject的两个方法:
willChangeValueForKey:
和
didChangeValueForKey:

在一个被观察属性发送改变之前,willChangeValueForKey:一定会被调用,这就会记录旧的值。而当改变发生后,observeValueForKey:ofObject:change:context:会被调用,继而didChangeValueForKey:也会被调用。如果手动实现这些,就可以实现“手动触发”了。

那么“手动触发”的使用场景是什么?一般我们只在希望能控制“回调的调用时机”时才会这么做。

具体做法如下:

如果这个value是表示时间的self.now,那么代码如下:最后两行代码缺一不可。

//  手动触发 value 的KVO,最后两行代码缺一不可。

//@property (nonatomic, strong) NSDate *now;

- (void)viewDidLoad {    

[super viewDidLoad];  

_now = [NSDat edate];                     [self addObserver:self forKeyPath:@"now" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"1");    [self willChangeValueForKey:@"now"];// “手动触发self.now的KVO”,必写。
NSLog(@"2");    [self didChangeValueForKey:@"now"];// “手动触发self.now的KVO”,必写。
NSLog(@"4");

}

但是平时我们一般不会这么干,我们都是等系统去“自动触发”。“自动触发”的实现原理:

 
比如调用setNow:时,系统还会以某种方式在中间插入willChangeValueForKey:、didChangeValueForKey:和observeValueForKeyPath:ofObject:change:contenxt:的调用。

大家可能以为这是因为setNow:是合成方法,有时候我们也能看到有人这么写代码:

 
- (void)setNow:(NSDate*)aDate {   

[self willChangeValueForKey:@"now"];// 没有必要_now = aDate;   

[self didChangeValueForKey:@"now"];// 没有必要

}



这完全没有必要,不要这么做,这样的话,KVO代码会被调用两次。KVO在调用存取方法之前总是调用willChangeValueForKey:,之后总是调用didChangeValueForKey:。












 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值