版本记录
版本号 | 时间 |
---|---|
V1.0 | 2017.09.14 |
前言
KVO
具有更强大的功能,是苹果给我们的一个回调机制,在某个对象注册监听者后,在被监听的对象发生改变时,对象会发送一个通知给监听者,以便监听者执行回调操作。接下来几篇就详细的解析一下KVO。感兴趣的可以看上面几篇。
1. KVO解析(一) —— 基本了解
2. KVO解析(二) —— 一个简单的KVO实现
KVO合规性
考虑到对于指定属性KVO的合规性,一个类必须满足下面关系:
改类对于指定的属性必须是 key-value coding compliant ,这在 Ensuring KVC Compliance中已经指明。
该类发送对于属性的
KVO change
通知。依赖的key被合适的注册(可以参考Registering Dependent Keys)
Automatic Change Notification
NSObject提供了一个自动键值改变通知的基本实现,自动键值更改通知通知观察者使用键值兼容的访问器进行的更改以及键值编码方法。 由例如mutableArrayValueForKey
返回的集合代理对象也支持自动通知。
下面例子展示的是name属性更改是观察者收到的通知。
//Examples of method calls that cause KVO change notifications to be emitted
// 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];
Manual Change Notification
有的时候,你也许想要控制通知的处理过程,例如,针对应用程序特定原因而不必要的触发通知,或将一些更改分组到单个通知中。 手动更改通知提供了这样做的手段。
手动和自动通知不是互斥的。 除了自动的通知之外,您还可以自由发布手动通知。 更通常地,您可能希望完全控制特定属性的通知。 在这种情况下,您将重写NSObjec的automaticNotificationsObserversForKey
的实现。 对于要排除其自动通知的属性,automaticNotificationsObserversForKey
的子类实现应该返回NO。一个子类实现应该为任何无法识别的key调用super超类。 下面代码中的示例启用了对balance
属性的手动通知,允许超类确定所有其他键的通知。
// Example implementation of automaticallyNotifiesObserversForKey:
+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)theKey
{
BOOL automatic = NO;
if ([theKey isEqualToString:@"balance"]) {
automatic = NO;
}
else {
automatic = [super automaticallyNotifiesObserversForKey:theKey];
}
return automatic;
}
为了实现手动观察通知,在值改变之前,你需要调用方法 willChangeValueForKey:,并在值改变之后调用方法 didChangeValueForKey:,下面的例子实现了对balance
属性的手动通知。
- (void)setBalance:(double)theBalance
{
[self willChangeValueForKey:@"balance"];
_balance = theBalance;
[self didChangeValueForKey:@"balance"];
}
您可以通过首先检查值是否已更改来最小化发送不必要的通知。 下面的例子展示的是balance
的值改变时,并且在已经改变的时候提供通知。
// Testing the value for change before providing notification
- (void)setBalance:(double)theBalance
{
if (theBalance != _balance) {
[self willChangeValueForKey:@"balance"];
_balance = theBalance;
[self didChangeValueForKey:@"balance"];
}
}
如果单个操作导致多个键更改,则必须嵌套更改通知,如下所示。
- (void)setBalance:(double)theBalance
{
[self willChangeValueForKey:@"balance"];
[self willChangeValueForKey:@"itemChanged"];
_balance = theBalance;
_itemChanged = _itemChanged+1;
[self didChangeValueForKey:@"itemChanged"];
[self didChangeValueForKey:@"balance"];
}
在有序的多关系的情况下,您不仅必须指定更改的密钥,还必须指定所涉及的对象的更改类型和索引。 更改类型是NSKeyValueChange
,它指定NSKeyValueChangeInsertion
,NSKeyValueChangeRemoval
或NSKeyValueChangeReplacement
。 受影响的对象的索引作为NSIndexSet
对象传递。
下面中的代码片段演示了如何在多对象关系事务中包装对象的删除。
// Implementation of manual observer notification in a to-many relationship
- (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"];
}
后记
未完,待续~~~