ios kvo理解

2018-11-14  本文已影响5人  pengmengli

KVO是key-value observing的缩写

KVO是Objective-C对观察者设计模式的实现

Apple使用isa混写(isa-swizzling)来实现KVO

通过isa混写就是,当我们注册一个观察者,调用- (void)addObserver:(NSObject*)observer forKeyPath:(NSString*)keyPath options:(NSKeyValueObservingOptions)options context:(nullablevoid*)context,去观察一个对象的属性,系统会在运行时动态为我们创建一个NSKVONotifying_A类,以前类的isa会指向新创建的NSKVONotifying_A类。NSKVONotifying_A就是原来类的子类,他会重写Setter方法,重写Setter方法会通知所有的观察者

下面我们代码实现一下

先创建两个类KObject和KObserver,通过名字我们可以看出两个类的作用

KObject.h

#import  <Foundation/Foundation.h>

@interfaceKObject :NSObject

/** 注释 */

@property (nonatomic,assign) int value;

- (void)add;

@end

KObject.m

#import "KObject.h"

@implementation KObject

- (id)init{

    self= [superinit];

    if(self) {

        _value=0;

    }

    return self;

}

- (void)add{

    _value+=1;

}

@end

KObserver.m

#import "KObserver.h"

#import "KObject.h"

@implementation KObserver

- (void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context{

    if([objectisKindOfClass:[KObjectclass]] && [keyPathisEqualToString:@"value"]) {

        NSNumber *number = [change valueForKey:NSKeyValueChangeNewKey];

        NSLog(@"value is %@",number);

    }

}

引用两个类

KObject*obj = [[KObjectalloc]init];

    KObserver*observer = [[KObserveralloc]init];

    NSLog(@"obj_old : %s ",object_getClassName(obj));

    [objaddObserver:observer forKeyPath:@"value" options:NSKeyValueObservingOptionNew context:nil];

    NSLog(@"obj_new : %s",object_getClassName(obj));

    obj.value=1;

打印结果

2018-11-14 17:02:20.890465+0800 CategrayTest[70728:13350625] obj_old : KObject

2018-11-14 17:02:20.890982+0800 CategrayTest[70728:13350625] obj_new : NSKVONotifying_KObject

2018-11-14 17:02:20.891274+0800 CategrayTest[70728:13350625] value is 1

我们可以看到未添加观察之前我们的obj属于KObject类,添加完Observer,属于NSKVONotifying_KObject类,这就是我们上面说的isa混写

我们刚才说系统为我们重写了Setter方法,具体内部怎么实现的的呢,我们看一下代码

- (void)setValue:(int)obj{

    [self willChangeValueForKey:@"keyPath"];

    [super setValue:obj];

    [self didChangeValueForKey:@"keyPath"];

}

从代码中我们可以看到,系统为我们重写Setter方法,内部调用了willChangeValueForKey和didChangeValueForKey,其实调用didChangeValueForKey会触发添加观察者类,去调用observeValueForKeyPath。。。这个方法,来达到我们观察一个对象的属性的目的

还有一个问题,我们怎么给value赋值,才能出发Setter方法,通过点可以触发,那么通过kvc可以吗,我们试一下

obj.value=1;

[obj setValue:@2 forKey:@"value"];

打印结果

2018-11-14 17:24:36.286900+0800 CategrayTest[71027:13409991] value is 1

2018-11-14 17:24:36.287082+0800 CategrayTest[71027:13409991] value is 2

看来是可以的

如果直接赋值呢

obj.value=1;

  [obj setValue:@2 forKey:@"value"];

   [obj add];

打印结果

2018-11-14 17:28:42.525141+0800 CategrayTest[71091:13421105] value is 1

2018-11-14 17:28:42.525303+0800 CategrayTest[71091:13421105] value is 2

看来不行,add方法实现,上面代码中有

如果我们在属性赋值之前和之后分别加上willChangeValueForKey和didChangeValueForKey呢

add方法改装

- (void)add{

    [self willChangeValueForKey:@"value"];

    _value+=1;

    [self didChangeValueForKey:@"value"];

}

    obj.value=1;

    [objsetValue:@2 forKey:@"value"];

    [objadd];

打印结果

2018-11-14 17:31:11.251041+0800 CategrayTest[71147:13428256] value is 1

2018-11-14 17:31:11.251222+0800 CategrayTest[71147:13428256] value is 2

2018-11-14 17:31:11.251510+0800 CategrayTest[71147:13428256] value is 3

看来我们可以手动实现kvo

上一篇下一篇

猜你喜欢

热点阅读