一、KVC
通过关键字,访问属性及成员变量。
1.kvc当调用setValue:属性值 forKey:@”name“的代码时,底层的执行机制(顺序)如下:
①.set方法:set<Key> (注意,通过属性自动合成的set方法也算)
(BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,继续向下寻找是否有满足的成员变量。
(如果你重写了该方法让其返回NO的话,那么在这一步KVC会直接执行⑤setValue:forUNdefinedKey:方法,不过一般开发者不会这么做。所以,如果想让自己的类禁用kvc,可以用此方法返回NO,不过记得属性会自动合成set方法)
②.成员变量:
_<key>成员变量,
_is<Key>成员变量,
<key>成员变量,
is<Key>成员变量,
③.执行该对象的setValue:forUNdefinedKey:方法,默认是抛出异常。
- 当调用valueForKey:@"name"时,执行机制(顺序)如下:
①.访问器方法:get<Key>,<key> ,is<Key>
②.成员变量:
_<key>成员变量,
_is<Key>成员变量,
<key>成员变量,
is<Key>成员变量。
所以KVC支持实例变量。
- KVC的keyPath中的集合运算符
@avg、@sum、@max、@min、@count
必须用在集合对象或集合属性上,@"@sum.age"或 @"集合属性.@max.age"。比如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))。