部分代码均可参考AFNetworking
文件
1.KVO监听属性(自动监听)
1.1.注册监听
AFURLSessionManager.m 156
[progress addObserver:self
forKeyPath:NSStringFromSelector(@selector(fractionCompleted))
options:NSKeyValueObservingOptionNew
context:NULL];
1.2.监听回调
AFURLSessionManager.m 171
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {
if ([object isEqual:self.downloadProgress]) {
if (self.downloadProgressBlock) {
self.downloadProgressBlock(object);
}
}
else if ([object isEqual:self.uploadProgress]) {
if (self.uploadProgressBlock) {
self.uploadProgressBlock(object);
}
}
}
1.3.移除监听(如果在对象监听对象释放的时候,监听没有移除,程序会奔溃)
- (void)dealloc {
[self.downloadProgress removeObserver:self forKeyPath:NSStringFromSelector(@selector(fractionCompleted))];
[self.uploadProgress removeObserver:self forKeyPath:NSStringFromSelector(@selector(fractionCompleted))];
}
2.KVO监听属性(手动监听)
手动监听与自动监听几乎一样,都有上面的三个步骤,只是,手动监听,需要实现下列方法
+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key
这个方法接收监听的key,返回YES表示使用自动监听,而返回NO则是手动监听,一般手动监听需要实现监听key的set方法,例如
AFURLRequestSerialization.m 266
- (void)setAllowsCellularAccess:(BOOL)allowsCellularAccess {
[self willChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
_allowsCellularAccess = allowsCellularAccess;
[self didChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
}
手动监听相对于自动监听有如下优点,可以避免不必要的通知,例如将上述代码修改如下
- (void)setAllowsCellularAccess:(BOOL)allowsCellularAccess {
if (_allowsCellularAccess != allowsCellularAccess)
{
[self willChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
// 此处增加需要执行的额外操作
_allowsCellularAccess = allowsCellularAccess;
[self didChangeValueForKey:NSStringFromSelector(@selector(allowsCellularAccess))];
}
}
当然利用手动监听,你还可以实现对数组和字典这样内部存储数据改变是后出发通知,不过不好的是,需要你在每个数组/字典改变得地方加上如下代码
[p willChangeValueForKey:NSStringFromSelector(@selector(array))];
[p.array addObject:@"123"];
[p didChangeValueForKey:NSStringFromSelector(@selector(array))];
也许有更好的方法,也可以给我留下言,互相交流
3.利用KVO实现某个属性因多个属性值得改变而改变
实现代码:
+ (NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key
{
if ([key isEqualToString:NSStringFromSelector(@selector(array))])
{
return [NSSet setWithObjects:NSStringFromSelector(@selector(age)),NSStringFromSelector(@selector(name)),nil];
}
return [super keyPathsForValuesAffectingValueForKey:key];
}
上面代码的意思是指,当age或者name发生改变时,会触发array的KVO通知,值得一提的NSSet
里面存储的是keyPaths
路径,也就是说你可以存储xxx.ooo
这样的路径进行监听
4.移除通知
在监听的对象里面移除监听
- (void)dealloc
{
[self removeObserver:self forKeyPath:NSStringFromSelector(@selector(age))];
[self removeObserver:self forKeyPath:NSStringFromSelector(@selector(name))];
[self removeObserver:self forKeyPath:NSStringFromSelector(@selector(array))];
}
5.注意点
对象实现了KVO
以后,系统会利用运行时向系统注册一个新的类,例如Person
实现了KVO
,那么系统会创建这样的一个类NSKVONotifying_Person
这样Person类会带来一个问题,在实现
+ (void)initialize
{
NSLog(@"-----%@",[self class]);
}
方法的时候,这个方法会被调用两次,所以如果你同时实现了这两个方法,需要用class进行判定,这样代码才不会重复调用