KVO学习

作者: AryCode | 来源:发表于2017-02-17 10:44 被阅读56次

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

    相关文章

      网友评论

        本文标题:KVO学习

        本文链接:https://www.haomeiwen.com/subject/pwchwttx.html