- (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
。bash
NSKeyValueObservingOptionNew 新值
NSKeyValueObservingOptionOld 旧值
NSKeyValueObservingOptionInitial 观察最初的值(在注册观察服务时会调用一次触发方法)
NSKeyValueObservingOptionPrior 分别在值修改先后触发方法(即一次修改有两次触发)
复制代码
context
:打上标签,在观察回调中可用于区分,防止继承。object
也能够达到相似的功能,但object
须要遍历不少东西,context
性能更好。不想用的话就传入NULL
。 object
:被观察的对象。打印出来是<Person: 0x600001815710>
。 change
:发生改变的内容。性能
####KVO的基本用法 1.建立一个类,命名Person
,分别添加成员变量与属性变量。ui
@interface Person : NSObject{
NSString *nickName;
}
@property(strong,nonatomic)NSString * name;
@property(assign,nonatomic)int age;
@property(strong,nonatomic)NSMutableArray * cars;
@end
复制代码
2.在控制器中,建立Person
对象,给对象添加观察者。atom
@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.添加观察回调spa
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"改变的内容===%@", change);
NSLog(@"观察的对象===%@", object);
}
复制代码
4.移除观察者(必须),不移除有可能会致使崩溃。3d
-(void)dealloc{
[self.p removeObserver:self forKeyPath:@"name"];
}
复制代码
5.触发监听(举例)code
-(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];
}
复制代码
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
//按照通常套路将不会回调
[self.p.cars addObject:@"兰博基尼"];
//KVO 中 ,对集合类型(Array,set)操做。
//取值和赋值过程 跟通常状况不同,须要经过KVC 的方式。
[[self.p mutableArrayValueForKey:@"cars"] addObject:@"兰博基尼"];
}
复制代码
因为KVO并无开源,为了了解KVO内部的实现原理,尝试 探索KVO底层原理orm