OC中的KVO

Posted talk_8

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了OC中的KVO相关的知识,希望对你有一定的参考价值。

看官们,我们在前面章回中介绍了OC中的KVC,本章回中将介绍OC中的KVO。

我们说的KVO是Key Value Observing的缩写,它主要用来监听类属性的变化,它是在NSObject类中实现的,通常情况下我们定义的类只需要继承NSObject类就可以使用KVO来监听类属性的变化。下面是KVO的使用方法:


1.给属性注册监听器


 

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


访方法用来给属性添加监听器,第一个参数observer表示属性所属的对象,通常我们写成self。第二个参数是属性的名称。第三个参数表示监听的事件,或者说发生什么事件后我们才能监听到消息,该参数是NSKeyValueObservingOptions类型的值,它是一个枚举类型,包含四个枚举常量,我们在这里不详细介绍它们,大家可以参考官方文档,通常情况下,我们使用NSKeyValueObservingOptionNew这个枚举常量,它表示属性值发生变化这个事件,或者说我们可以监听到属性值发生变化后的事件。第四个参数是context,我们不需要关注,通常写成nil就可以。


2.重写监听器方法


-

 (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSKeyValueChangeKey, id> *)change
                       context:(void *)context;


该方法是系统提供的方法,当被监听的事件发生后系统会调用该方法,我们重写它时只需要把事件发生后程序需要做的事情写到此方法中就可以。

第一个参数是被监听的属性名称;第二个参数是属性所属的对象;第三个参数是一个字典,字典中的key是框架提供的NSString类型的常量,我们直接拿来使用就可以,常用的是NSKeyValueChangeNewKey这个常量,它表示被监听属性值变化后的key,通过此Key可以在字典change中获取到被监听属性值变化后的value.通俗点说就是属性的新值。除了这个常量外,框架还提供了其它四个NSString类型的常量,我们在这里就不一一介绍发,大家参考官方文档就可以。需要注意的是这个常量需要和注册监听器时NSKeyValueObservingOptions类型的枚举常量匹配,不然获取不到相关的值。比如想知道属性被修改前的值,那么在注册监听器时需要使用NSKeyValueObservingOptionOld枚举常量,然后通过NSKeyValueChangeOldKey从change这个字典中获取属性被修改前的值。大家在实践时可以打开代码中被注释掉的代码来做验证。第四个参数暂时不需要关注;


3.给属性注销监听器


- (void)removeObserver:(NSObject *)observer
            forKeyPath:(NSString *)keyPath;


访方法用来注销属性的监听器,第一个参数observer表示属性所属的对象,通常我们写成self。第二个参数是属性的名称。


介绍完KVO的操作步骤和具体的方法后,接下来通过具体的代码来演示如何使用KVO。此代码是在KVC示例代码的基础上写的,代码中添加了相关的注释以方便大家理解代码。

#include<Foundation/Foundation.h>

//定义一个名叫Person的类并且实现该类
@interface Person : NSObject
@property NSString *name;
@property int age;

-(void) print;
-(void) addObserver;
-(void) removeObserver;

@end

@implementation Person
@synthesize name;
@synthesize age;

-(void) print 

	NSLog(@"Person name: %@, age: %d",name,age);


//给name属性注册监听器
-(void) addObserver

	[self addObserver:self forKeyPath:@"name"
		options:NSKeyValueObservingOptionNew context:nil];
	// [self addObserver:self forKeyPath:@"name"
		// options:NSKeyValueObservingOptionOld context:nil];
	NSLog(@"add Observer for key: name");


//重写监听属性的方法,当属性变化时回调该方法
- (void)observeValueForKeyPath:(NSString *)keyPath 
                      ofObject:(id)object 
                        change:(NSDictionary<NSKeyValueChangeKey, id> *)change 
                       context:(void *)context 

	NSString *oldValue = [change objectForKey:NSKeyValueChangeOldKey];
	NSString *newValue = [change objectForKey:NSKeyValueChangeNewKey];

	NSLog(@"KVO key: %@, old value: %@ ,new value: %@",keyPath,oldValue,newValue);


//注销属性的监听器
-(void) removeObserver

	[self removeObserver:self forKeyPath:@"name"];
	NSLog(@"remove observer for key: name");

@end

int main()

	Person * p1 = [[Person alloc] init];
	Person * p2 = [[Person alloc] init];

	//在修改属性前注册监听器
	[p1 addObserver];

	//通过setter方法给类的属性赋值,通过getter方法获取类的属性值
	[p1 setName:@"ZhangSan"];
	[p1 setAge:26];
	[p1 print];
	[p1 setName:@"WangWu"];
	NSLog(@"p1: name: %@, age: %d",[p1 name],[p1 age]);

	//属性修改完成后注销监听器。可以在注册和注销监听器之间查看是否可以监听到属性变化的消息
	[p1 removeObserver];

	//p2对象没给属性没有给属性注册监听器,不会监听到属性变化的消息;
	//通过KVC方法给类的属性赋值,并且获取类的属性值
	[p2 setValue:@"LiSi" forKey:@"name"];
	[p2 setValue:@"LiSi" forKey:@"name"];
	[p2 setValue:[NSNumber numberWithInt:33] forKey:@"age"];
	[p2 print];
	NSLog(@"p2: name: %@, age: %@",[p2 valueForKey:@"name"],[p2 valueForKey:@"age"]);

我们在代码中新定义了两个方法,用来注册和注销监听器,同时重写了监听器的方法,这三个方法就是刚才介绍KVO三个步骤的体现,在main方法中则是对KVO三个步骤的综合应用。我们在这里就不列出程序运行结果了希望大家自己动手去实践。


最后我们对本章回的内容做一个总结:

  • 1.只需要继承NSObject类就可以使用KVO;
  • 2.使用KVO时主要包含注册、重写、注销监听器三个步骤;
  • 3.只有注册了监听器的属性才能监听到属性变化的消息;
  • 4.注册监听器时使用的事件类型与监听器中获取属性值使用的事件类型(字典中的key)需要匹配;

看官们,本章回的内容就介绍到这里,欲知后事如何且听下回分解! 

以上是关于OC中的KVO的主要内容,如果未能解决你的问题,请参考以下文章

细说OC中的KVO

OC常用设计模式机制之KVO

[OC学习笔记]KVO原理

[OC学习笔记]KVO原理

61 (OC)* 代理 block 通知 代理 kvo

OC中的内存管理