1,KVC(键值编码) Key Value Coding
1.1在C#中,可以通过字符串反射来获取对象,从而对对象的属性进行读写,Object-C中有同样的实现,通过字符串(属性名词)对对象的属性进行读写。
1.2KVC的操作方法由NSKeyValueCoding协议来定义,而Object对象实现该协议,所以所有的对象都具有KVC功能
动态设置:setValue:属性值 forKey:属性名 setValue: 属性值forKeyPath:属性路径
动态读取:valueForKey:属性名 valueForKeyPath:属性路径
名词解释:属性名(简单路径) 属性路径(复合路径,比如A对象有一个b属性,那么A.b就是复合属性)
1.3KVC动态设置规则:(假设对a进行设置)首先查找setA方法,如果没有setA方,则查找_a变量,如果没有_a变量,则查找a变量,如果没有a变量,则调用 setValue: forUndefinedKey:方法。对于方法和属性,不管公有还是私有都可以设置成功
1.4KVC动态读取规则:(假设对a进行读取)首先读取a方法,如果没有a方法,则查找_a变量,如果没有_a变量,则查找a变量,如果没有a变量,则调用valueForUndefinedKey:方法。对于方法和属性,不管公有还是私有都可以设置成功
1.5 KVC 与 对象的读写方法有什么区别
1,可以通过属性的名字直接访问属性,而不是调用的属性的accssor。
2,减少代码量
for example:场景一:
People *people = [peoleArray objectAtIndex:row];
if ([[column identifier] isEqualToString:@"name"]) {
return [people name];
}
if ([[column identifier] isEqualToString:@"age"]) {
return [people age];
}
利用KVC : return [people valueForKey:[colum identifier]];
场景二:可以把一个NSDictionary 的元素直接赋值给对象的属性。
-(id) initWithDictionary:(NSMutableDictionary*) jsonObject
{
if((self = [super init]))
{
[self init];
[self setValuesForKeysWithDictionary:jsonObject];
}
return self;
}
setValuesForKeysWithDictionary: 把NSDictionary中key对应的对象属性设置上NSDictionary中key对应的value.
异常处理:(1)如果NSDictionary中key 与对象中属性的名称不一致时,我们需要 setValue: forUndefinedKey: 方法来处理没有匹配上的属性,否则setValuesForKeysWithDictionary:将抛出异常(NSUndefinedKeyException)
(2)如果NSDictionay 内嵌的结构比较复杂:Products{product1{count:xx, sumPrice:xx}}, product2{} ….},我们需要重写 setValue: forKey: 方法。
- -(void) setValue:(id)value forKey:(NSString *)key
- {
- if([key isEqualToString:@"products"])
- {
- for(NSMutableDictionary *productDict in value)
- {
- Prodcut *product = [[Product alloc] initWithDictionary:prodcutDict];
- [self.products addObject:product];
- }
- }
- }
场景三: 我们需要把数组里People对象的name名字的首字母大写,调用kvc:
return [array valueForKeyPath:@"name.capitalizedString"] (这里是valueForKeyPath)
1.6 KVC集合运算符
插入一个知识点:NSNumber 封装的数据类型有 Char,UnsignedChar, Short, UnsignedShort, Int, UnsignedInt, Long, UnsignedLong, LongLong, UnsignedLongLong, Float, Double, Bool, Integer, UnsingedInteger.
NSValue 封装的数据类型:CGPoint, CGSize, CGRect, CGAffineTransform, UIEdgeInsets, UIOffset.
集合运算符是一组特殊的key path,只能传递给 valueForKeyPath: 该方法。
集合运算符的格式:
共支持5种运算:@sum @avg @count @max @min
2,KVO(键值监听) Key Value Observing (观察者模式)
2.1KVO的操作方法由NSKeyValueObserving协议来指定,NSObject对象实现了该协议,所以所有的对象都拥有KVO操作。
2.2KVO操作的方法如下:
注册指定key路径的监听器: addObserver: forKeyPath: options: context:
删除指定key路径的监听器:removeObserver: forKeyPath:、 removeObserver: forKeyPath: context:
监听器回调方法: observeValueForKeyPath: ofObject: change: contenxt:
options的值说明:
NSKeyVlaueObservingOptionNew
需要补充。。。。。。。
change: 参数为NSDictionary , key 为 new 和 old
2.3KVO 与NSNotificationCenter的区别
好处:1,代码更简洁,不需要修改被观察的类,永远都是观察者在做事情
不足:1,如果没有observer 监听 keyPath, removeOberser: forKeyPath: 会crash。
2,很难定位是谁监听了对象的属性
2.4 KVO的自动触发与手动触发
自动触发原理:键值观察通知依赖于NSObject的两个方法:willChangeValueForKey: 和 didChangeVlaueForKey:。在一个被观察属性发生改变之前,调用willChangeValueForKey,改变之后调用didChangeVlaueForKey,然后调用 observerValueForKey: ofObject: change: context:.
模拟一个被观察属性发生改变场景:
- (void)setNow:(NSDate *)aDate {
[self willChangeValueForKey:@
"now"
];
// 运行时,自动添加方法
_now = aDate;
[self didChangeValueForKey:@
"now"
];
// 运行时,自动添加方法
}
实现原理, isa-Swizzing.
原理解释: 当我们监听一个对象时,一个新类会被创建,新类继承自原来对象的类,并重写了被观察属性的setter方法,(运行时,加入了自动触发属性变更的通知),最后通过isa-Swizzing,把这个对象的isa指针,指向新创建的子类。
该图来自(ios程序猿)