key-value observing(KVO)

来源

https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/KeyValueObserving/KeyValueObserving.html

key-value observing 是一种机制,当一些对象的指定属性被改变时,通知另一些对象.

这个机制对于App里的model和controller之间的联系是非常有用的.一个controller对象通常都会观察models对象的属性,并且view对象也会通过controller对象观察model对象的属性.另外,一个model对象也会观察另一个依赖的model对象的属性.
你可以观察普通的属性,to-one关系,to-many关系(数组等等).to-many的观察会被不同的改变类型而被通知,比如在当前改变类型中,哪些对象会被需要(插入,删除,替换等)



注册 KVO

你必须执行以下步骤来确保对象能接收KVO通知:

  • 使用addObserver:forKeyPath:options:context:.方法,把观察者注册到被观察者里.
  • 在观察者对象里实现observeValueForKeyPath:ofObject:change:context:方法,去接收改变通知.
  • 当不再需要接收信息时,用removeObserver:forKeyPath:方法去除注册.至少观察者在被内存释放前你要调用这个方法.

KVO的addObserver:forKeyPath:options:context:方法不会保持强引用到观察对象,被观察对象或context.你应该保证你自己保持这些的强引用


如果一个notification被传到最顶层,NSObject就会抛出一个NSInternalInconsistencyException,那是因为这是一个编程错误:一个注册在子类的通知被漏了,传到了NSObject.在observeValueForKeyPath:ofObject:change:context:都会实现super


移除一个观察者,要注意以下几点:

  • 移除一个未注册的观察会引用NSRangeException异常.要不在removeObserver:forKeyPath:context:调用前,确保调用了addObserver:forKeyPath:options:context:.或者你把removeObserver:forKeyPath:context:放在 try/catch里去处理异常.
  • 你一个观察对象不会在对象释放时,自动移除自己.被观察对象还会继续发送通知,不会知道观察者的状态.这样,一个通知就有可能会发送一个已释放的对象,引用一个内存访问异常.所以,你要确保在观察者被释放之前它们要移除自己.
  • 通常的模式是在观察者初始化时注册通知(比如:init 或viewDidLoad),并且在释放时移除通知(比如:deallocdeinit).保证它们成对的顺序的增加移除通知.

自动改变通知

NSObject提供了自动改变通知的基础实现.以下代码都会实现自动通知:

// Call the accessor method.
[account setName:@"Savings"];
 
// Use setValue:forKey:.
[account setValue:@"Savings" forKey:@"name"];
 
// Use a key path, where 'account' is a kvc-compliant property of 'document'.
[document setValue:@"Savings" forKeyPath:@"account.name"];
 
// Use mutableArrayValueForKey: to retrieve a relationship proxy object.
Transaction *newTransaction = <#Create a new transaction for the account#>;
NSMutableArray *transactions = [account mutableArrayValueForKey:@"transactions"];
[transactions addObject:newTransaction];

手动操控改变通知

在一些情况,你可能想要控制一个处理的通知,比如最少的不必要的通知,或把一组通知集合在一个通知里.
你要重写NSObject的类方法automaticallyNotifiesObserversForKey:.
以下代码把"balance"的通知给排除了.

+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)theKey {
 
    BOOL automatic = NO;
    if ([theKey isEqualToString:@"balance"]) {
        automatic = NO;
    }
    else {
        automatic = [super automaticallyNotifiesObserversForKey:theKey];
    }
    return automatic;
}
    class override func automaticallyNotifiesObservers(forKey key:String) -> Bool{
        <#Code#>
    }

为了实现手动通知,你要在改变对象之前调用willChangeValueForKey:,并且在改变对象之后调用didChangeValueForKey:.

- (void)setBalance:(double)theBalance {
    if (theBalance != _balance) {
        [self willChangeValueForKey:@"balance"];
        _balance = theBalance;
        [self didChangeValueForKey:@"balance"];
    }
}

如果一个操作引起来了多个keys的变化,你要去嵌套改变通知:

- (void)setBalance:(double)theBalance {
    [self willChangeValueForKey:@"balance"];
    [self willChangeValueForKey:@"itemChanged"];
    _balance = theBalance;
    _itemChanged = _itemChanged+1;
    [self didChangeValueForKey:@"itemChanged"];
    [self didChangeValueForKey:@"balance"];
}

如果是在一个有序的to-many的关系(比如:Array),你不仅要指出key,还要指出改变的类型和被改地方的索引.

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

推荐阅读更多精彩内容