键值编码-KVC

KVC全称是Key Value Coding,允许开发者通过key直接访问对象的属性,对对象的属性进行赋值。可以在运行时动态的访问和修改对象的属性。


  • KVC最常用的两个方法也是最基本的操作方法:
  1. setValue: forKey:为指定的属性设置值
  2. valueForKey:获取指定属性的值。

  • KVC的简单用法
    定义一个Person类
#import 
@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end

@implementation Person

- (void)setName:(NSString *)name {
    _name = name;
    NSLog(@"setName被调用");
}

@end

然后通过KVC的方式给Person类的属性赋值:

Person *person = [[Person alloc] init];
[person setValue:@"张三" forKey:@"name"];
[person setValue:[NSNumber numberWithInt:11] forKey:@"age"];
NSLog(@"person.name=%@",[person valueForKey:@"name"]);
NSLog(@"person.age=%@",[person valueForKey:@"age"]);

控制台输出如下:

键值编码-KVC_第1张图片
屏幕快照 2019-01-02 21.28.40.png

可以看到对Person类的属性赋值成功了。并且 - (void)setName:(NSString *)name方面被调用了。
将代码修改为下面这样:

@interface Person : NSObject {
    NSString *_name;
    NSString *name;
}
//@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end

name赋值:

Person *obj = [[Person alloc] init];
[obj setValue:@"小明" forKey:@"name"];

通过查看obj实例可以看出是_name被赋值了。

键值编码-KVC_第2张图片
屏幕快照 2019-01-02 21.43.54.png

将代码更改为

@interface Person : NSObject {
    NSString *_isName;
    NSString *name;
}
//@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end

调用代码不变,可以看到_isName被赋值了

键值编码-KVC_第3张图片
屏幕快照 2019-01-02 21.45.53.png

将代码修改为

@interface Person : NSObject {
    NSString *isName;
    NSString *name;
}
//@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end

再次运行,结果为name被赋值。

键值编码-KVC_第4张图片
屏幕快照 2019-01-02 21.49.40.png


setValue: forKey:的底层实现机制:

[person setValue:@"张三" forKey:@"name"]来举例说明。

  1. 先寻找Person类的setName方法,使用setter方法完成赋值。
  2. 如果没有找到setName:方法,KVC机制会检查+ (BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,如果你重写了该方法让其返回NO的话,那么在这一步KVC会执行setValue:forUndefinedKey:方法
  3. 如果Person类中没有setName方法,KVC会依次去搜索_name,_isName,name,isName成员变量去进行赋值操作。无论这些成员变量在什么地方定义和什么控制符修饰,都会进行赋值操作。
  4. 如果上述三条都没找到,会调用setValue:forUndefinedKey:的方法,抛出异常。

在上面的例子中再增加一条代码,对Person类中没有的属性进行赋值。

[person setValue:@"dog" forKey:@"dog"];

运行代码,看到控制台报错

屏幕快照 2017-12-25 下午10.06.47.png

可以看到调用了 setValue:forUndefinedKey:方法,程序结束。如果这不是我们想要的结果,我们可以重写这个方法。

- (void)setValue:(id)value forUndefinedKey:(NSString *)key{
    NSLog(@"你尝试设置的key=%@不存在",key);
}

这样就可以避免程序因异常而结束。


valueForKey:的底层实现机制:
[person valueForKey:@"name"]来举例说明。

  1. 先寻找Person类的name方法,使用getter方法完成取值。
  2. KVC机制会检查+ (BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,如果你重写了该方法让其返回NO的话,那么在这一步KVC会执行valueForUndefinedKey:方法
  3. 如果Person类中没有name方法,KVC会依次去搜索_name,_isName,name,isName成员变量去进行取值。无论这些成员变量在什么地方定义和什么控制符修饰,都会进行取值。
  4. 如果上述三条都没找到,会调用valueForUndefinedKey:的方法,抛出异常。

在上面的例子中再增加一条代码,对Person类中没有的属性进行赋值。

NSLog(@"person.dog=%@",[person valueForKey:@"dog"]);

控制台打印:

屏幕快照 2017-12-25 下午10.20.00.png

可以看到调用了 valueForUndefinedKey:的方法,程序因异常结束。同样可以重写 valueForUndefinedKey:来避免程序异常结束。

你可能感兴趣的:(键值编码-KVC)