iOS学习笔记iOS Developer

KVO学习

2017-02-17  本文已影响56人  AryCode

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

上一篇下一篇

猜你喜欢

热点阅读