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底层原理