招聘一个靠谱的iOS KVC KVO 45-46-48-49-51

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

// 添加键值观察
/*
1 观察者,负责处理监听事件的对象
2 观察的属性
3 观察的选项
4 上下文
*/
[self.person addObserver:self 
              forKeyPath:@"name" 
                 options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld 
                 context:@"Person Name"];

observer中需要实现该方法:

// 所有的 kvo 监听到事件,都会调用此方法
/*
 1. 监听的被观察者的属性键路径
 2. 被观察者对象
 3. change 被观察者对象属性变化字典(新/旧)
 4. 上下文,与监听的时候传递的一致
 */
- (void)observeValueForKeyPath:(NSString *)keyPath 
                      ofObject:(id)object 
                        change:(NSDictionary *)change
                       context:(void *)context;

46. 如何手动触发一个value的KVO

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

自动触发 KVO 的原理

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

willChangeValueForKey: 

didChangevlueForKey: 

在一个被观察属性发生改变之前, willChangeValueForKey: 一定会被调用,这就 会记录旧的值。而当改变发生后,

observeValueForKey:ofObject:change:context:

会被调用,继而 didChangeValueForKey: 也会被调用。如果可以手动实现这些调用,就可以实现“手动触发KVO”了。

//
//  ViewController.m
//  46
//
//  Created by zhaoyingxin on 16/8/22.
//  Copyright © 2016年 zhaoyingxin@aliyun.com. All rights reserved.
//

#import "ViewController.h"

@interface ViewController ()
@property (nonatomic, strong) NSDate *now;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self manualKVO];
}

- (void)manualKVO{
    [self addObserver:self
           forKeyPath:@"now"
              options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew
              context:@"self.now.kvo"];
    
    NSLog(@"1");
    [self willChangeValueForKey:@"now"]; // “手动触发self.now的KVO”,必写。
    NSLog(@"2");
    [self didChangeValueForKey:@"now"]; // “手动触发self.now的KVO”,必写。
    NSLog(@"4");
}

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSString *,id> *)change
                       context:(void *)context {
    NSLog(@"3");
    
    NSLog(@"keyPath = %@",keyPath);
    NSLog(@"object  = %@",object);
    NSLog(@"change  = %@",change);
    NSLog(@"context = %@",context);
}

//2016-08-22 11:45:45.805 46[44377:802802] 1
//2016-08-22 11:45:45.805 46[44377:802802] 2
//2016-08-22 11:45:45.806 46[44377:802802] 3
//2016-08-22 11:45:45.806 46[44377:802802] keyPath = now
//2016-08-22 11:45:45.806 46[44377:802802] object  = <ViewController: 0x7fb33a6317c0>
//2016-08-22 11:45:45.806 46[44377:802802] change  = {
//    kind = 1;
//    new = "<null>";
//    old = "<null>";
//}
//2016-08-22 11:45:45.806 46[44377:802802] context = self.now.kvo
//2016-08-22 11:45:45.807 46[44377:802802] 4

@end

“自动触发”的实现原理:
比如调用 setNow: 时,系统还会以某种方式在中间插入 willChangeValueForKey:didChangeValueForKey:observeValueForKeyPath:ofObject:change:context: 的调用。

- (void)setNow:(NSDate *)aDate {
    [self willChangeValueForKey:@"now"]; // 没有必要
    _now = aDate;
    [self didChangeValueForKey:@"now"];  // 没有必要
}

这完全没有必要,不要这么做,这样的话,KVO代码会被调用两次。KVO在调用存取方法之前总是调用 willChangeValueForKey:,之后总是调用 didChangeValueForkey:。怎么做到的呢?答案是通过 isa 混写(isa-swizzling)

参考苹果官方文档:
https://developer.apple.com/library/ios/documentation/Cocoa/Conceptual/KeyValueObserving/Articles/KVOCompliance.html#//apple_ref/doc/uid/20002178-SW3

48.KVC的keyPath中的集合运算符如何使用?

1.必须用在集合对象上或普通对象的集合属性上
2.简单集合运算符有@avg, @count , @max , @min ,@sum,
3.格式 @"@sum.age"或 @"集合属性.@max.age"

49. KVC和KVO的keyPath一定是属性么?

KVO支持实例变量

51.apple用什么方式实现对一个对象的KVO?

Apple 的文档对 KVO 实现的描述:

Automatic key-value observing is implemented 
using a technique called 
isa-swizzling... 
When an observer is registered for an attribute of an object 
the isa pointer of the observed object is modified, 
pointing to an intermediate class rather than at the true class ...

从Apple 的文档可以看出:Apple 并不希望过多暴露 KVO 的实现细节。不过,要是借助 runtime 提供的方法去深入挖掘,所有被掩盖的细节都会显示:

当给一个对象添加观察者时,一个新的类会被动态创建。
这个类继承自该对象原本的类,并重写了被观察属性的 setter 方法。
重写的 setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察者对象:属性值的更改。
最后通过 isa 混写(isa-swizzling) 把这个对象的 isa 指针 
( isa 指针告诉 Runtime 系统这个对象的类是什么 ) 
指向这个新创建的子类,对象就神奇的变成了新创建的子类的实例。
KVO实现原理.png
Apple 使用了 isa 混写(isa-swizzling)来实现 KVO
KVO.png

didChangeValueForKey: 注释掉, observeValueForKeyPath:ofObject:change:context:不会执行

observeValueForKeyPath:ofObject:change:context: 是在 didChangeValueForKey: 内部触发的操作

调用顺序
willChangeValueForKey:---> didChangeValueForKey:--->
observeValueForKeyPath:ofObject:change:context:

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

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

推荐阅读更多精彩内容