El principio subyacente de iOS: la esencia de KVC

El nombre completo de KVC es Key-Value Coding, comúnmente conocido como "clave-valor de codificación", que puede acceder a un atributo a través de una clave.

1. Uso común de la API

-(void)setValue:(id)valor paraKeyPath:(NSString *)keyPath;
-(void)setValue:(id)valor forKey:(NSString *)key;
-(id)valueForKeyPath:(NSString *)keyPath;
-(id)valorParaClave:(NSString *)clave;

Cree dos nuevas clases YMPerson y YMCat, el código es el siguiente:

#import <Foundation/Foundation.h>
#import "YMCat.h"

NS_ASSUME_NONNULL_BEGIN

@interface YMPerson : NSObject
@property (nonatomic, assign) int age;

@property (nonatomic, strong) YMCat *cat;
@end

NS_ASSUME_NONNULL_END
复制代码
#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface YMCat : NSObject
@property (nonatomic, assign) int weight;
@end

NS_ASSUME_NONNULL_END
复制代码

1. La diferencia entre setValue:forKey:ysetValue:forKeyPath:

Establecer la edad de YMPerson y el nombre de YMCat

YMPerson *person = [[YMPerson alloc] init];
person.cat = [[YMCat alloc] init];

//通过kvc修改age属性
[person setValue:@11 forKey:@"age"];
NSLog(@"person.age = %d",person.age);
[person setValue:@22 forKeyPath:@"age"];
NSLog(@"person.age = %d",person.age);

//setValue@"" forKey@""只能设值第一层的属性
//  [person setValue:@"少点" forKey:@"name"]; //报错

    [person.cat setValue:@"少点" forKey:@"name"];
    NSLog(@"cat.name = %@",person.cat.name); //少点

//  setValue@"" forKeyPath@"" 
//  Path就是路径 可以一级一级的寻找就行赋值或者取值操作
    [person setValue:@"多点" forKeyPath:@"cat.name"];
    NSLog(@"cat.name = %@",person.cat.name); //多点
复制代码

2. Escenarios comunes en el proyecto

1) operación de interfaz de usuario

[textField setValue:[UIColor grayColor] forKeyPath:@"_placeholderLabel.textColor"];
复制代码

Barra de pestañas personalizada: puede personalizar un objeto UITabbar y luego crear la vista que desea internamente y volver a diseñarla internamente a través del método layoutSubviews. Luego, a través de KVC, reemplace la propiedad de la barra de pestañas de UITabbarController con una clase personalizada.

2) Diccionario para modelar

Status *status = [[self alloc] init];
//利用KVC字典转模型
[status setValuesForKeysWithDictionary:dict];    
return status;
复制代码

En el caso de la conversión de diccionario a modelo, si asigna valores uno por uno en el método de inicio personalizado, debe cambiar la instrucción de asignación cada vez que cambien los datos. Sin embargo, a través de la API de asignación proporcionada por KVC, los datos se pueden asignar por lotes. setValuesForKeysWithDictionary:Supongamos que hay los siguientes datos JSON y la clase Estado está definida, y el Estado es asignado por métodos en el mundo exterior .

3) Usar con KVO

根据 KVO的本质分析,其是在运行时生成新的子类并重写setter方法,在其内容发生改变时发送消息。但是这只是对属性直接进行赋值会触发,如果属性时容器对象,对容器对象进行add或remove操作,则不会调用KVO的方法。可以通过KVC对应的API来配合使用,使容器内部发生改变的时候也能触发KVO。
在进行容器对象操作时,先带哦用下面方法通过key或者keyPath获取集合对象,然后再对容器对象进行add或remove等操作时,就会触发KVO的消息通知了。

Key方法

- (NSMutableArray *)mutableArrayValueForKey:(NSString *)key;
- (NSMutableOrderedSet *)mutableOrderedSetValueForKey:(NSString *)key API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (NSMutableSet *)mutableSetValueForKey:(NSString *)key;
复制代码

KeyPath方法:

- (NSMutableArray *)mutableArrayValueForKeyPath:(NSString *)keyPath;
- (NSMutableOrderedSet *)mutableOrderedSetValueForKeyPath:(NSString *)keyPath API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (NSMutableSet *)mutableSetValueForKeyPath:(NSString *)keyPath;
复制代码

代码验证:

[self.person1 addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:@"context-age"];
[self.person1 addObserver:self forKeyPath:@"mutArr" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:@"context-mutArr"];
复制代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    //非容器属性修改 setter方法 会触发KVO监听
    [_person1 setValue:@33 forKey:@"age"];
    
    //直接修改容器属性 不会触发KVO
//    [_person1.mutArr addObject:@"add"];

    NSMutableArray *mutArr = [_person1 mutableArrayValueForKey:@"mutArr"];
    [mutArr addObject:@"add"];
}
复制代码

二、KVC赋值原理

1.按照 setKey:, _setKey: 顺序查找方法,找到了就调用方法传递参数。
2.第一步没找到就会调用 accessInstanceVariablesDirectly 方法,该方法返回值为NO时直接调用 setValue:forUndefinedKey: 并抛出异常NSUnknownKeyException,方法返回值是YES的时候进入第三步。该方法默认值是返回YES。
3.按照_key、_isKey、key、isKey顺序查找成员变量,找到了就直接赋值,没找到依然是调用setValue:forUndefinedKey:并抛出异常NSUnknownKeyException

流程示意图:

KVC赋值原理图.jpg

三、KVC取值原理

1.kvc取值按照 getKey、key、iskey、_key 顺序查找方法
存在直接调用
2.没找到同样,先查看accessInstanceVariablesDirectly方法
如果return YES; > 可以直接访问成员变量
如果return NO; > 不可以直接访问成员变量,
3.如果可以访问会按照 _key、_isKey、key、iskey的顺序查找成员变量
找到直接复制
未找到报错NSUnkonwKeyException错误

四、总结

1、通过KVC修改属性值 是否会触发KVO?

答:会触发。因为KVC也是走了setKey方法

2、通过KVC修改成员变量值,是否会触发KVO?

答:会触发。因为KVC内部会调用willChangeValueForKey和didChangeValueForKey方法

3、直接修改成员变量会触发KVO吗?

答:不会。直接修改成员变量,不会走setKey方法

[person setValue:@11 forKey:@"age"]; //能触发
person->age = 11111; //不能触发
复制代码

4、字典setValue和setObject区别

setValue: forKey:

  • 1、方法是KVC(键-值编码),方法是在NSObject对象中创建的,也就是说所有的oc对象都有这个方法,所以可以 用于任何类.
YMPerson *person = [[YMPerson alloc] init]; 
//当对象有name属性的时候就是通过KVC来赋值 
[person setValue:@"mystrict" forKey:@"name"];
复制代码
  • 2、value可以是nil,如果为nil,则自动调用removeObject forkey方法
  • 3、key必须为string类型的
  • 4、取值的时候valueforkey的key如果包含@符号,则取值时会自动把@去掉,程序crash

setObject: forKey:

  • 1、方法是NSMutableDictionary特有的;
  • 2、value不可以为nil,如果为nil,则程序崩溃;但value可以为[NSNull null], [NSNull null]为一个空对象,并不是nil;
  • 3、Key的对象是一个id类型,并不是NSString,只不过我们经常使用NSString而已。
  • 4、取值的时候objectforkey的key如果包含@符号,无影响,可以正常取出该值

Supongo que te gusta

Origin juejin.im/post/7083414587995291656
Recomendado
Clasificación