KVO学习

部分代码均可参考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进行判定,这样代码才不会重复调用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 在编程中,最常见的就是程序的流程取决于你所使用的各种变量和属性的值,根据变量和属性的值确定后面运行的代码,有时会检...
    pro648阅读 1,656评论 2 27
  • kvo 一、kvo使用 1.添加观察者 给某个需要监听的对象添加一个观察者。 各个参数的意义是: observer...
    cr7aa阅读 299评论 0 0
  • 本文结构如下: Why? (为什么要用KVO) What? (KVO是什么) How? ( KVO怎么用) Mo...
    等开会阅读 1,663评论 1 21
  • 本文由我们团队的 纠结伦 童鞋撰写。 文章结构如下: Why? (为什么要用KVO) What? (KVO是什么...
    知识小集阅读 7,427评论 7 105
  • 看起来有点粗糙,继续练习! 线条!线条!
    一笑堂LY阅读 192评论 0 0