注意:KVC是是对对象的属性进行操作的
我们知道在C#中可以通过反射读写一个对象的属性,有时候这种方式特别方便,因为你可以利用字符串的方式去动态控制一个对象。其实由于ObjC的语言特性,你不必进行任何操作就可以进行属性的动态读写,这种方式就是Key Value Coding(简称KVC)。
KVC的操作方法由NSKeyValueCoding协议提供,而NSObject就实现了这个协议,也就是说ObjC中几乎所有的对象都支持KVC操作,常用的KVC操作方法如下:
- 动态设置:
setValue:属性值 forKey:属性名
(用于简单路径)、setValue:属性值 forKeyPath:属性路径
(用于复合路径,例如Person有一个Account类型的属性,那么person.account就是一个复合属性) - 动态读取:
valueForKey:属性名
、valueForKeyPath:属性名
(用于复合路径)
下面通过一个例子来理解KVC
Account.h
#import <Foundation/Foundation.h>
@interface Account : NSObject
@property (nonatomic,assign) float balance;
@end
Account.m
#import "Account.h"
@implementation Account
@end
Person.h
#import <Foundation/Foundation.h>
@class Account;
@interface Person : NSObject{
@private
int _age;
}
@property (nonatomic,copy) NSString *name;
@property (nonatomic,retain) Account *account;
-(void)showMessage;
@end
Person.m
#import "Person.h"
@implementation Person
-(void)showMessage{
NSLog(@"name=%@,age=%d",_name,_age);
}
@end
main.m
#import <Foundation/Foundation.h>
#import "Person.h"
#import "Account.h"
int main(int argc, const char * argv[]) {
@autoreleasepool {
Person *person1=[[Person alloc]init];
[person1 setValue:@"Kenshin" forKey:@"name"];
[person1 setValue:@28 forKey:@"age"];//注意即使一个私有变量仍然可以访问
[person1 showMessage];
//结果:name=Kenshin,age=28
NSLog(@"person1's name is :%@,age is :%@",person1.name,[person1 valueForKey:@"age"]);
//结果:person1's name is :Kenshin,age is :28
Account *account1=[[Account alloc]init];
person1.account=account1;//注意这一步一定要先给account属性赋值,否则下面按路径赋值无法成功,因为account为nil,当然这一步骤也可以写成:[person1 setValue:account1 forKeyPath:@"account"];
[person1 setValue:@100000000.0 forKeyPath:@"account.balance"];
NSLog(@"person1's balance is :%.2f",[[person1 valueForKeyPath:@"account.balance"] floatValue]);
//结果:person1's balance is :100000000.00
}
return 0;
}
KVC使用起来比较简单,但是它如何查找一个属性进行读取呢?具体查找规则(假设现在要利用KVC对a进行读取):
- 如果是动态设置属性,则优先考虑调用setA方法,如果没有该方法则优先考虑搜索成员变量_a,如果仍然不存在则搜索成员变量a,如果最后仍然没搜索到则会调用这个类的setValue:forUndefinedKey:方法(注意搜索过程中不管这些方法、成员变量是私有的还是公共的都能正确设置);
- 如果是动态读取属性,则优先考虑调用a方法(属性a的getter方法),如果没有搜索到则会优先搜索成员变量_a,如果仍然不存在则搜索成员变量a,如果最后仍然没搜索到则会调用这个类的valueforUndefinedKey:方法(注意搜索过程中不管这些方法、成员变量是私有的还是公共的都能正确读取);