首先认识一个概念:
响应式编程: 简称RP(Reactive Programming)
响应式编程是一种面向数据流和变化传播的编程范式。这意味着可以在编程语言中很方便地表达静态或动态的数据流,而相关的计算模型会自动将变化的值通过数据流进行传播。
OC中的通知、代理、KVO、事件(addTarget)同属于响应式,添加观察者,设置代理,添加Target类似于响应式编程中的 “订阅”
KVO
_p = [[Person alloc]init];
// 添加观察者,当name属性值发生变化就会通知self去响应
[_p addObserver:self forKeyPath:@”name” options:NSKeyValueObservingOptionNew context:nil];
// 响应观察
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
}
原理: OC的属性是成员变量 _name 跟 getter 、setter 的封装。
改变成员变量的值 p->_name = @"xiaowu"; 不会响应方法,
所以KVO内部(不只是)是观察setter方法。
底层原理:①利用 runtime 动态创建一个 Person 子类 NSKVONotifing_Person ,重写 setName 方法
-(void)setName:(NSString*)name{
[self willChangeValueForKey:@”name”];
[super setName:name];
[self didChangeValueForKey:@”name”];
/* 以上两个change方法会触发响应,options参数传
NSKeyValueObservingOptionNew新值则会在改变之后
didChange触发,old在改变之前willChange触发
*/
}
② 动态改变 P 对象的类型(一个对象的真实类型,调用方法看 isa 指针,其内部是改变isa指针所指向的类),变为子类类型,所以当用 setter 改变属性值时 _P.name=@”xiaowu”; 会调用子类重写的 setName 方法,进行响应
KVO分为两种触发模式:自动触发(一改变值就自动触发响应);手动触发(变化不一定每次都通知,有时候需要满足一定条件之后再通知)
在被观察的 Person 类 .m 文件里面添加类方法:
+(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key{
if([key isEqualToString:@”name”]){
return NO;
}
return YES;// YES自动触发,NO为手动触发
}
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[self willChangeValueForKey:@"name"];
_p.name = @"xiaowu";
[self didChangeValueForKey:@"name"];//当改变后就会手动触发
}
KVO中属性的依赖关系:dog 是 _p 的属性,dog 中有 age 、level 属性,可直接通过 keyPath:dog.age 观察 age 的改变;
当我们要观察 dog 中的 age 、level 属性时得 addObserve 两次,比较麻烦,可以为其添加依赖关系,只需观察 dog ,然后在 Person 类中实现类方法:
+(NSSet<NSString *> *)keyPathsForValuesAffectingValueForKey:(NSString *)key{
NSSet *keyPaths = [super keyPathsForValuesAffectingValueForKey:key];
if ([key isEqualToString:@"dog"]) {
// 属性依赖关系里面的属性名称一定要有下划线成员变量
NSArray *affectingKeys = @[@"_dog.age",@"_dog.level"];
keyPaths = [keyPaths setByAddingObjectsFromArray:affectingKeys];
}
return keyPaths;
}
KVO观察不到集合属性的变化?(如 _p 的数组属性 arr ,往 arr 中 addObject 添加元素时不会响应),因为KVO是监听 setter 方法,所以不会响应。
可以借助KVC:
//根据 *\_p* 对象的 *arr* 属性返回一个新的数组
NSMutableArray*tmpArr = [_p mutableArrayValueForKey:@"arr"];
[tmpArr addObject:@”11”];
打印响应的字典 change 得到
kind 有四种,如下图:
默认为1,监听setter,2为插入,3为删除,4为替换;2、3、4都为容器类型的属性监听,其内部原理跟setter一样,也是动态的生成容器的子类,重写addObject、remove、replace等方法,改变isa指针指向子类。
函数式编程:如AFN的Block设计,响应式中混合函数式,回调在一块block中,相比代理、通知等业务逻辑更加紧密。