iOS KVO

作者: 尤先森 | 来源:发表于2019-03-07 17:51 被阅读0次

    KVO主要的几个方法

    • 添加观察者
    - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
    
    • 移除观察者
    - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
    
    • 观察回调
    - (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context;
    

    参数说明:
    observer:观察者,一般传self
    keyPath:路径,传入观察对象的属性变量,(成员变量不会生效,因为成员变量没有setter方法)。
    options:枚举,一共有4个值,最常用的为NSKeyValueObservingOptionNew

        NSKeyValueObservingOptionNew       新值
        NSKeyValueObservingOptionOld       旧值
        NSKeyValueObservingOptionInitial   观察最初的值(在注册观察服务时会调用一次触发方法)
        NSKeyValueObservingOptionPrior     分别在值修改前后触发方法(即一次修改有两次触发)
    

    context:打上标签,在观察回调中可用于区分,防止继承。object也可以达到类似的功能,但object需要遍历很多东西,context性能更好。不想用的话就传入NULL
    object:被观察的对象。打印出来是<Person: 0x600001815710>
    change:发生改变的内容。

    KVO的基本用法

    1.创建一个类,命名Person,分别添加成员变量与属性变量。

    @interface Person : NSObject{
        NSString *nickName;
    }
    @property(strong,nonatomic)NSString * name;
    @property(assign,nonatomic)int age;
    @property(strong,nonatomic)NSMutableArray * cars;
    @end
    

    2.在控制器中,创建Person对象,给对象添加观察者。

    @property(strong,nonatomic)Person *  p;
    
    self.p = [[Person alloc]init];
    [self.p addObserver:self forKeyPath:@"name" options:(NSKeyValueObservingOptionNew) context:NULL];
    [self.p addObserver:self forKeyPath:@"age" options:(NSKeyValueObservingOptionNew) context:NULL];
    

    3.添加观察回调

    -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
        NSLog(@"改变的内容===%@", change);
        NSLog(@"观察的对象===%@", object);
    }
    

    4.移除观察者(必须),不移除有可能会导致崩溃。

    -(void)dealloc{
        [self.p removeObserver:self forKeyPath:@"name"];
    }
    

    5.触发监听(举例)

    -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
        self.p.name = @"尤先森";
        self.p.age ++;
        [[self.p mutableArrayValueForKey:@"cars"] addObject:@"阿斯顿马丁"];
    }
    

    一些骚操作

    • 自动观察
      观察对象设置自动观察(与其说是自动,不如说是可以加限制)
      下面这个方法只有key等于name的时候,才会触发观察回调。
    +(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key{
        if ([key isEqualToString:@"name"]) {
            return YES;
        }
        return NO;
    }
    
    • 手动观察
      当被上面的方法限制了的话,可以通过手动观察让他恢复观察
    -(void)setAge:(int)age{
        [self willChangeValueForKey:@"age"];
        _age = age;
        [self didChangeValueForKey:@"age"];
    }
    
    • 联动观察
      当我们需要观察一个对象受有多个因素影响时,可进行联动操作。
    // 1.新建3个属性
    @property(assign,nonatomic)double done; //已完成
    
    @property(assign,nonatomic)double total; //总量
    
    @property(strong,nonatomic)NSString *downloadProgress; //下载进度
    
    // 2.关联属性
    +(NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key{
        NSSet *keyPaths = [super keyPathsForValuesAffectingValueForKey:key];
        if ([key isEqualToString:@"downloadProgress"]) {
            NSArray *array = @[@"done",@"total"];
            keyPaths = [keyPaths setByAddingObjectsFromArray:array];
        }
        return keyPaths;
    }
    
    // 3.添加监听
    [self.p  addObserver:self forKeyPath:@"downloadProgress" options:(NSKeyValueObservingOptionNew) context:NULL];
    
    // 4.触发
    -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{    
        self.p.done +=10;
    }
    
    //downloadProgress懒加载
    -(NSString *)downloadProgress{
        if (self.done == 0) {
            _done = 10;
        }
        if (self.total == 0) {
            _total = 100;
        }
        return [NSString stringWithFormat:@"%f",1.0f*self.done/self.total];
    }
    
    
    • 对集合类型(NSMutableArray,NSMutableSet)进行操作。
    -(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
        //按照一般套路将不会回调
        [self.p.cars addObject:@"兰博基尼"];
        //KVO 中 ,对集合类型(Array,set)操作。
        //取值和赋值过程 跟一般情况不一样,需要通过KVC 的方式。
        [[self.p mutableArrayValueForKey:@"cars"] addObject:@"兰博基尼"];
    }
    

    扩展

    由于KVO并没有开源,为了了解KVO内部的实现原理,尝试
    探索KVO底层原理

    相关文章

      网友评论

          本文标题:iOS KVO

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