iOS相关知识(三)--- KVO相关

1、什么是KVO

KVO的全称是Key-Value Observing,俗称“键值监听”,可以用于监听某个对象属性值的改变

2、如何使用KVO

第一步:创建一个Person类 然后添加一个name属性

//.h文件
#import 

NS_ASSUME_NONNULL_BEGIN

@interface Person : NSObject
@property (nonatomic,strong)NSString *name;

@end

NS_ASSUME_NONNULL_END

//.m文件
#import "Person.h"

@implementation Person

@end

第二步 在控制器中初始化这个Person

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


@interface ViewController ()
@property (nonatomic,strong)Person *person1;
@property (nonatomic,strong)Person *person2;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.person1 = [[Person alloc] init];
    self.person1.name = @"张三";
    
    self.person2 = [[Person alloc] init];
    self.person2.name = @"王五";
    
    //给person1的name添加监听
    NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld;
    [self.person1 addObserver:self forKeyPath:@"name" options:options context:@"这是名字的标记"];
    
    /**
     注意:context 其实就是一个标示符,如果一个对象需要监听多个属性值的时候,
     可以使用context来区分是哪一个值发生了改变。
     可以使用字符串来定义当前那个属性发生了改变,
     - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
     方法中的context 就是从添加监听的时候传过来的。如添加监听的时候context 传的是:这是名字的标记
     那么在触发函数中接收的context也是:这是名字的标记
     */
}

#pragma mark -- 当self.person.name 发生改变的时候就会触发这个方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context{
    NSLog(@"新值:%@--旧值:%@--标记:%@",change[@"new"],change[@"old"],context);
}

#pragma mark -- 点击屏幕改变name
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    self.person1.name = @"李四";
    self.person2.name = @"赵六";
}

#pragma mark -- 当控制器销毁的时候直接移除监听器
- (void)dealloc{
    [self.person1 removeObserver:self forKeyPath:@"name"];
}

@end

如上代码即完成了self.person1.name的监听

3、KVO底层实现

代码如2中的代码,创建两个person的实例变量 person1和person2,使用KVO对person1的name进行监听,
点击屏幕的时候 打印 person1和person2 的isa 结果如图所示


person1和person2的isa.png

如上图所示:

person1的isa 指向的是 NSKVONotifying_Person
person2的isa 指向的是 Person

疑问:

实例对象的isa指向的类对象,怎么person1的isa指向的却是 NSKVONotifying_Person  这个是什么玩意?

区别:

person1 与 person2 唯一的区别就是 person1添加了KVO监听 而person2没有被监听。

结论:

 1、当实例对象 进行KVO观察时候,会利用RuntimeAPI动态生成一个子类,然后将对象的isa指向新生成的子类
 2、KVO本质上是监听属性的setter方法,只要被观察对象有成员变量和对应的set方法,
就会调用Foundation的_NSSetValueAndNotify函数这个函数内部会执行 willChangeVlaueForKey函数、
父类的setter方法 和didChangeVlaueForKey的方法
(didChangeVlaueForKey 方法内部会触发监听器的observeValueForKeyPath: ofObject: context:函数
 3、子类会重写父类的set、class、dealloc、_isKVOA方法
 4、当观察对象移除所有的监听后,会将观察对象的isa指向原来的类
 5、当观察对象的监听全部移除后,动态生成的类不会注销,而是留在下次观察时候再使用,避免反复创建中间子类

person1 isa 指针指向如下图所示


person1 isa 指针指向.png

person2 isa 指针指向如下图所示


person2 isa 指针指向.png

4、如何手动出发KVO

由KVO的底层原理可以看出 触发KVO主要就是 runtime动态生成的子类重写了set方法 在set方法中 
首先调用Foundation的_NSSetValueAndNotify函数这个函数内部会执行 willChangeVlaueForKey函数
父类的setter方法
 didChangeVlaueForKey的方法
在didChangeVlaueForKey方法调用的时候就会触发监听器的
observeValueForKeyPath: ofObject: context:函数

所以想要手动触发KVO 就必须手动调用willChangeVlaueForKey 和didChangeVlaueForKey方法

5、直接修改成员变量会不会触发KVO

KVO 的本质就是重写属性的setter方法,如果直接修改成员变量就根本没有去触发setter方法 
所以直接修改成员变量并不会触发KVO

你可能感兴趣的:(iOS相关知识(三)--- KVO相关)