KVO底层原理

首先认识一个概念:

响应式编程: 简称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 *)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底层原理_第1张图片
img.png

KVO分为两种触发模式:自动触发(一改变值就自动触发响应);手动触发(变化不一定每次都通知,有时候需要满足一定条件之后再通知)

在被观察的 Person.m 文件里面添加类方法:

+(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key{
    if([key isEqualToString:@”name”]){
        return NO;
    }
    return YES;// YES自动触发,NO为手动触发
}
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    [self willChangeValueForKey:@"name"];
    _p.name = @"xiaowu";
    [self didChangeValueForKey:@"name"];//当改变后就会手动触发
}

KVO中属性的依赖关系:dog_p 的属性,dog 中有 agelevel 属性,可直接通过 keyPath:dog.age 观察 age 的改变;

当我们要观察 dog 中的 agelevel 属性时得 addObserve 两次,比较麻烦,可以为其添加依赖关系,只需观察 dog ,然后在 Person 类中实现类方法:

+(NSSet *)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 ,往 arraddObject 添加元素时不会响应),因为KVO是监听 setter 方法,所以不会响应。
可以借助KVC:

//根据 *\_p* 对象的 *arr* 属性返回一个新的数组
NSMutableArray*tmpArr = [_p mutableArrayValueForKey:@"arr"];
[tmpArr addObject:@”11”];

打印响应的字典 change 得到

KVO底层原理_第2张图片
img1.png

kind 有四种,如下图:

KVO底层原理_第3张图片
img2.png

默认为1,监听setter,2为插入,3为删除,4为替换;2、3、4都为容器类型的属性监听,其内部原理跟setter一样,也是动态的生成容器的子类,重写addObject、remove、replace等方法,改变isa指针指向子类。

函数式编程:如AFN的Block设计,响应式中混合函数式,回调在一块block中,相比代理、通知等业务逻辑更加紧密。

你可能感兴趣的:(KVO底层原理)