iOS-KVC/KVO

一、KVC

通过关键字,访问属性及成员变量。
1.kvc当调用setValue:属性值 forKey:@”name“的代码时,底层的执行机制(顺序)如下:
①.set方法:set (注意,通过属性自动合成的set方法也算)

(BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,继续向下寻找是否有满足的成员变量。
(如果你重写了该方法让其返回NO的话,那么在这一步KVC会直接执行⑤setValue:forUNdefinedKey:方法,不过一般开发者不会这么做。所以,如果想让自己的类禁用kvc,可以用此方法返回NO,不过记得属性会自动合成set方法)

②.成员变量:
_成员变量,
_is成员变量,
成员变量,
is成员变量,
③.执行该对象的setValue:forUNdefinedKey:方法,默认是抛出异常。

  1. 当调用valueForKey:@"name"时,执行机制(顺序)如下:
    ①.访问器方法:get ,is
    ②.成员变量:
    _成员变量,
    _is成员变量,
    成员变量,
    is成员变量。

所以KVC支持实例变量。

  1. KVC的keyPath中的集合运算符
    @avg、@sum、@max、@min、@count
    必须用在集合对象或集合属性上,@"@sum.age"或 @"集合属性[email protected]"。比如Person类中有个age属性,
NSArray *array = [NSArray arrayWithObjects:p1, p2, p3, nil];
NSInteger max = [[array valueForKeyPath:@"@max.age"] integerValue];

二、KVO

1. 基本使用

KVO是一种机制,对象可以指定一个观察者,来观察它的某个属性,当该属性发生改变时,观察者会收到通知,来执行相应的操作。

只需两部即可完成:
步骤一、 添加键值观察

[self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"Person Name"];

参数:

  • 观察者,负责处理监听事件的对象
  • 观察的属性
  • 观察的选项
  • 上下文

步骤二、observer中需要实现一下方法:
// 所有的 kvo 监听到事件,都会调用此方法

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

参数:

  • 观察的属性
  • 观察的对象
  • change 属性变化字典(新/旧)
  • 上下文,与监听的时候传递的一致

2. 如何手动触发KVO

键值观察通知依赖于 NSObject 的两个方法: willChangeValueForKey:didChangevlueForKey:。所以我们只需要直接调用这两个方法,即可手动触发KVO。

3. 手动实现实例变量支持KVO

如果将一个对象设定成属性,这个属性是自动支持KVO的,如果这个对象是一个实例变量,那么,这个KVO是需要我们自己来实现的

需要两个步骤:
步骤一:实现set方法,在set方法中添加手动触发KVO需要的两个方法。

-(void)setName:(NSString *)name
{
    [self willChangeValueForKey:@"name"];
    _name = name;
    [self didChangeValueForKey:@"name"];
}

步骤二:在该对象中实现automaticallyNotifiesObserversForKey方法,指定该key值为非自动监听

+(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key
{
    if ([key isEqualToString:@"name"]) {
        return NO;
    }
    return YES;
}

4. KVO的实现原理

当对象设置了观察者时,会生成一个新类,这个类继承自该对象的原类,这个新类会重写set方法,在set方法中增加KVO所依赖的两个方法willChangeValueForKey:didChangevlueForKey:方法,然后该对象的isa指针指向新生成的类(isa 混写(isa-swizzling))。

你可能感兴趣的:(iOS-KVC/KVO)