响应式编程思想-KVO底层分析-KVO实现

1.响应式编程思想

不需要考虑调用顺序,只需要考虑结果,类似于蝴蝶效应,产生一个事件,会影响很多东西,这些事件像流一样的传播出去,借用面向对象的一句话:万物皆是流

int a=3; int b=4; int c=a+b;
int a=0; int b=0; int c=a+b; a=3; b=4;

响应式编程思想:不考虑顺序,只考虑结果。变量 a b 值改变就会影响到 c 。变量 c 与变量 a b 的值绑定,只要变量a 或者b 的值发生改变 c 的值就发生改变;时刻要监听 a b 值的改变,改变了要马上响应 c

2.OC中响应式编程思想的使用 KVO时刻监听对象的属性变化

Person.h

#import <Foundation/Foundation.h>

@interface Person : NSObject
@property (nonatomic,assign) int age;
@end

ViewController.m

#import "ViewController.h"
#import "Person.h"

@interface ViewController()
@property (nonatomic,strong) Person *p;
@end

@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    
    Person *p = [[Person alloc] init];
    _p = p;
    
    [p addObserver:self
        forKeyPath:@"age"
           options:NSKeyValueObservingOptionNew
           context:nil];
}

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSString *,id> *)change
                       context:(void *)context{
    NSLog(@"%d",_p.age);
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    _p.age++;
}
@end

//2016-08-14 19:26:06.176 ZYXKVO[27499:1646080] 1
//2016-08-14 19:26:10.236 ZYXKVO[27499:1646080] 2
//2016-08-14 19:26:10.752 ZYXKVO[27499:1646080] 3
//2016-08-14 19:26:11.136 ZYXKVO[27499:1646080] 4

Person对象的年龄改变,就会被观察者self控制器观察到,就会调用观察者的 observeValueForKeyPath 方法通知观察者
只要Person的age发生改变马上就有响应,这就是 响应式编程思想

3.KVO底层实现机制

_p.age++;

KVO底层实现:就是判断有没有调用对象的set方法

1-对象p的isa指针是Person.jpg
2-对象p添加观察者后isa指针变为NSKVONotifying_Person.jpg
# KVO底层实现过程:
# 1>给一个对象添加观察者对象,会动态创建 "NSKVONotifying_该对象名" 的一个对象,"NSKVONotifying_Person" 是 "Person" 的子类
# 2>修改当前对象p的isa指针指向 "NSKVONotifying_Person"
# 3>只要调用对象p的set方法,就会改为调用 "NSKVONotifying_Person" 的set方法,因为对象p的isa指针改变了
# 4>重写 "NSKVONotifying_Person" 的set方法 : 1.[super set:] 2.通知观察者对象的属性改变

4.模仿KVO实现,实现响应式编程,运行时机制

3-对象p的isa指针是Person.jpg
4-将对象p的isa指针在运行时改为ZYXKVONotifying_Person.jpg

NSObject+ZYXKVO.h

#import <Foundation/Foundation.h>

@interface NSObject (ZYXKVO)

- (void)zyx_addObserver:(NSObject *)observer
             forKeyPath:(NSString *)keyPath
                options:(NSKeyValueObservingOptions)options
                context:(nullable void *)context;

@end

NSObject+ZYXKVO.m

#import "NSObject+ZYXKVO.h"

#import "ZYXKVONotifying_Person.h"
#import <objc/runtime.h>

@implementation NSObject (ZYXKVO)
- (void)zyx_addObserver:(NSObject *)observer
             forKeyPath:(NSString *)keyPath
                options:(NSKeyValueObservingOptions)options
                context:(void *)context{
// KVO底层实现过程:
// 1>给一个对象添加观察者对象,会动态创建 "NSKVONotifying_该对象名" 的一个对象,"NSKVONotifying_Person" 是 "Person" 的子类
// 2>修改当前对象p的isa指针指向 "NSKVONotifying_Person"
// 3>只要调用对象p的set方法,就会改为调用 "NSKVONotifying_Person" 的set方法,因为对象p的isa指针改变了
// 4>重写 "NSKVONotifying_Person" 的set方法 : 1.[super set:] 2.通知观察者对象的属性改变
    
    
    // 分类是不能有属性的,在运行时动态给分类设置属性
    // 修改isa指针,本质就是改变当前对象的类名
    object_setClass(self, ZYXKVONotifying_Person.class);
    
    // 把观察者对象保存为当前对象的一个属性
    // 运行时给 self 对象添加一个属性名为 @"observer" 的属性 observer
    // 引用策略为 nonatomic retain 强引用
    // 运行时给self创建一个强引用属性关联,self强引用观察者observer对象
    objc_setAssociatedObject(self, @"observer", observer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end

ZYXKVONotifying_Person.h

#import "Person.h"

@interface ZYXKVONotifying_Person : Person

@end

ZYXKVONotifying_Person.m

#import "ZYXKVONotifying_Person.h"
#import <objc/runtime.h>

@implementation ZYXKVONotifying_Person
- (void)setAge:(int)age{
    [super setAge:age];
    
    // 对象p调用了age的set方法就通知观察者 p.age 值改变了
    id observer = objc_getAssociatedObject(self, @"observer");
    // 调用观察者的方法
    [observer observeValueForKeyPath:@"age" ofObject:self change:nil context:nil];
}
@end

ViewController.m

#import "ViewController.h"

#import "Person.h"
#import "NSObject+ZYXKVO.h"

@interface ViewController()
@property (nonatomic,strong) Person *p;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    Person *p = [[Person alloc] init];
    _p = p;

    [p zyx_addObserver:self
            forKeyPath:@"age"
               options:NSKeyValueObservingOptionNew
               context:nil];
}

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSString *,id> *)change
                       context:(void *)context{
    NSLog(@"%d",_p.age);
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    _p.age++;
}
@end

//2016-08-14 20:16:17.529 ZYXKVO[32416:1685663] 1
//2016-08-14 20:16:18.340 ZYXKVO[32416:1685663] 2
//2016-08-14 20:16:18.500 ZYXKVO[32416:1685663] 3
//2016-08-14 20:16:18.788 ZYXKVO[32416:1685663] 5
//2016-08-14 20:16:18.925 ZYXKVO[32416:1685663] 6

这样就实现了KVO机制

[p zyx_addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.OC里用到集合类是什么? 基本类型为:NSArray,NSSet以及NSDictionary 可变类型为:NS...
    轻皱眉头浅忧思阅读 1,396评论 0 3
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,217评论 30 472
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,993评论 19 139
  • 一、深复制和浅复制的区别? 1、浅复制:只是复制了指向对象的指针,即两个指针指向同一块内存单元!而不复制指向对象的...
    iOS_Alex阅读 1,441评论 1 27
  • 《左耳》——人格裂变的N种可能 《左耳》是一部以校园、青春为主题的影片,由苏有朋执导,根据饶雪漫原创同名小说改编。...
    e4b685b8fdc5阅读 990评论 0 1