iOS KVO

2019-03-07  本文已影响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:@"阿斯顿马丁"];
}

一些骚操作

+(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];
}

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    //按照一般套路将不会回调
    [self.p.cars addObject:@"兰博基尼"];
    //KVO 中 ,对集合类型(Array,set)操作。
    //取值和赋值过程 跟一般情况不一样,需要通过KVC 的方式。
    [[self.p mutableArrayValueForKey:@"cars"] addObject:@"兰博基尼"];
}

扩展

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

上一篇下一篇

猜你喜欢

热点阅读