@interface People: NSObject @property (nonatomic, strong) NSString *name; @property (nonatomic, strong) NSNumber *age; @end
场景1,apple 官网的一个例子,当我们需要统计很多People的时候,每一行是一个人的实例,并且有2列属性,name, age, 这时候我们可以会这样做,
- (id)tableView:(NSTableView *)tableview objectValueForTableColumn:(id)column row:(NSInteger)row { People *people = [peoleArray objectAtIndex:row]; if ([[column identifier] isEqualToString:@"name"]) { return [people name]; } if ([[column identifier] isEqualToString:@"age"]) { return [people age]; } // And so on. }
同样我们也可以用KVC,帮助我们化简这些if, 因为name, age其实都是property, 我们可以直接通过key来访问,所以整理过后是
People *people = [peopleArray objectAtIndex:row]; return [people valueForKey:[column identifier]];
场景2,这下我们有了server, server的某个api(listPeople??), 会返回我们json格式一个数组,里面包含这样dict{name:xx, age:xx}这样的数据, 我们希望用这些dict数据构造出我们的people来,通常我们的做法是,为我们People类写一个static factory方法专门用来处理dict来, 把dict里面的数据取出来, 然后创建个空的People对象,然后依次设置property。然而当这样类似People的与server交互的类多了,我们就要为每个类都要加上这样的wrapper, 是否有种简单办法来设置这样的属性,当然就是我们的KVC了。
-(id) initWithDictionary:(NSMutableDictionary*) jsonObject { if((self = [super init])) { [self init]; [self setValuesForKeysWithDictionary:jsonObject]; } return self; }
setValuesForKeysWithDictionary, 会为我们把和dictionary的key名字相同的class proerty设置上dict中key对应的value, 是不是很方便呀,但是有同学又要问了 如果json里面的某些key就是和object的property名字不一样呢,或者有些server返回的字段是objc保留字如”id”, “description”等, 我们也希望也map dict to object, 这时候我们就需要用上setValue:forUndefinedKey, 因为如果我们不处理这些Undefined Key,还是用setValuesForKeysWithDictionary就会 抛出异常。
- (void)setValue:(id)value forUndefinedKey:(NSString *)key { if([key isEqualToString:@"nameXXX"]) self.name = value; if([key isEqualToString:@"ageXXX"]) self.age = value; else [super setValue:value forKey:key]; }
所以只要重载这个方法,就可以处理了那些无法跟property相匹配的key了,默认的实现是抛出一个NSUndefinedKeyException,又有同学发问了如果 这时候server返回的People有了内嵌的json(如Products{product1{count:xx, sumPrice:xx}}, product2{} ….),又该怎么办,能把这个内嵌的json转化成我们的客户端的Product类嘛, 当然可以这时候就需要重载setValue:forKey, 单独处理”Products”这个key, 把它wrapper成我们需要的class
-(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]; } } }
场景3,我们需要把一个数组里的People的名字的首字母大写,并且把新的名字存入新的数组, 这时候通常做法会是遍历整个数组,然后把每个People的name取出来,调用 capitalizedString 然后把新的String加入新的数组中。 有了KVC就有了新做法:
[array valueForKeyPath:@"name.capitalizedString"]
- (void)removeObservation { [self.object removeObserver:self forKeyPath:self.property]; } - (void)addObservation { [self.object addObserver:self forKeyPath:self.property options:0 context:(__bridge void*)self]; } - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { if ((__bridge id)context == self) { // 只处理跟我们当前class的property更新 } else { [super observeValueForKeyPath:keyPath ofObject:object change:change context:context]; } }
转载自:http://www.cocoachina.com/industry/20140224/7866.html
关于kvo与kvc参考:
http://yulingtianxia.com/blog/2014/05/12/objective-czhong-de-kvche-kvo/