8 iOS中KVO 的本质

 

 

前言本质

 

Automatic key-value observing is implemented using a technique called isa-swizzling

这计划的意思就是
自动的键值观察的实现基于 isa-swizzling

 

原理

1.KVO是基于runtime机制实现的

2.当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,

在这个派生类中重写基类中任何被观察属性的setter 方法。

派生类在被重写的setter方法内实现真正的通知机制

3.如果原类为Dog,那么生成的派生类名为NSKVONotifying_Dog

4.每个类对象中都有一个isa指针指向当前类,当一个类对象的第一次被观察,那么系统会偷偷将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter方法

5.键值观察通知依赖于NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey:一定会被调用,这就 会记录旧的值。而当改变发生后,didChangeValueForKey:会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。

 

 

 

一  KVO 的使用

 


#import "ViewController.h"
#import "Person.h"
@interface ViewController ()

@property (nonatomic,strong)Person *p1;

@property (nonatomic,strong)Person *p2;



@end

@implementation ViewController

- (void)viewDidLoad {
	[super viewDidLoad];
	// Do any additional setup after loading the view.
	
		// 监听person 类的属性值
	
	self.p1 = [[Person alloc]init];
	self.p1.name = @"p1_dog";
	
	
	self.p2 = [[Person alloc]init];
	self.p2.name = @"p2_dog";
	
	NSKeyValueObservingOptions val = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
	
	[self.p1 addObserver:self forKeyPath:@"name" options:val context:nil];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
	
	self.p1.name = @"p1_cate";
	self.p2.name = @"p2_cate";
	
	
}


- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
     	
	NSLog(@"%@ value changed from %@",keyPath,change);
}



@end

 

二 如何实现监听 ?

 

本质原因就是这两个对象的isa 指向不一样吗,主要是使用runtime 动态生成了一个派生类

 

1 如果未使用kvo 监听对象的时候
 

 

使用KVO 监听的话

 

 

三 代码验证

 


利用runtime 进行验证一下子,进入断电模式打印一下

 

	self.p1 = [[Person alloc]init];
	self.p1.name = @"p1_dog";
	
	self.p2 = [[Person alloc]init];
	self.p2.name = @"p2_dog";
	
	// 打印添加监听之前 setName的方法
	NSLog(@"监听之前%p--%p",[self.p1 methodForSelector:@selector(setName:)],[self.p1 methodForSelector:@selector(setName:)]);
	
	NSKeyValueObservingOptions val = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
	
	[self.p1 addObserver:self forKeyPath:@"name" options:val context:nil];
		
	NSLog(@"监听之后%p--%p",[self.p1 methodForSelector:@selector(setName:)],[self.p1 methodForSelector:@selector(setName:)]);
	
	NSLog(@"end");

添加之后,setName 的实现走的是_NSSetObjectValueAndNotify 方法

 

 

 

四 KVO的 调用顺序

 

KVO 的调用顺序是:

  • 调用 willChangeValueForKey:
  • 调用原来的 setter 实现
  • 调用 didChangeValueForKey:

也就是说 didChangeValueForKey: 内部必然是调用了 observerobserveValueForKeyPath:ofObject:change:context:方法。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值