iOS

iOS-KVC(三)内部取值深层次原理

2019-06-23  本文已影响0人  厦门_小灰灰

iOS-KVC(一)基本使用
iOS-KVC(二)内部赋值深层次原理
iOS-KVC(三)内部取值深层次原理
iOS-KVC(四)常见异常处理
iOS-KVC(五)容器类
iOS-KVC(六)正确性验证
iOS-KVC(七)字典相关
iOS-KVC(八)常见使用

主要涉及到的这个方法

- (nullable id)valueForKey:(NSString *)key;

KVC对key的搜索方式不同于之前说的赋值。
当用以上方法时候,底层的执行机制如下

  1. 首先按-get<Key>, -<key>, -is<Key> 这个顺序查找相关方法,找到的话会直接调用。
    如果返回值的类型是对象的话,就直接返回;
    如果是BOOL或者Int等值类型, 会将其包装成一个NSNumber对象返回;
    其他的就包装成NSValue对象返回。

  2. 如果上面的getter没有找到,KVC则会查找

-countOf<Key>, -objectIn<Key>AtIndex:或-<key>AtIndexes:

格式的方法。如果countOf<Key>方法和另外两个方法中的一个被找到,那么就会返回一个可以响应NSArray所有方法的代理集合(它是NSKeyValueArray,是NSArray的子类),调用这个代理集合的方法,或者说给这个代理集合发送属于NSArray的方法,就会以

countOf<Key>,-objectIn<Key>AtIndex:或-<key>AtIndexes:

这几个方法组合的形式调用。还有一个可选的get<Key>:range:方法。所以你想重新定义KVC的一些功能,你可以添加这些方法,需要注意的是你的方法名要符合KVC的标准命名方法,包括方法签名。

  1. 如果上面的方法没有找到,那么会同时查找
-countOf<Key>,-enumeratorOf<Key>, -memberOf<Key>:

格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所的方法的代理集合,和上面一样,给这个代理集合发NSSet的消息,就会以

-countOf<Key>,-enumeratorOf<Key>, -memberOf<Key>:

组合的形式调用。

  1. 如果还没有找到,再检查类方法
+ (BOOL)accessInstanceVariablesDirectly;

如果返回YES(默认行为),那么和先前的设值一样,会按_<key>, _is<Key>, <key>, is<Key>的顺序搜索成员变量名,这里不推荐这么做,因为这样直接访问实例变量破坏了封装性,使代码更脆弱。如果重写了类方法+ (BOOL)accessInstanceVariablesDirectly返回NO的话,那么会直接调用

- (id)valueForUndefinedKey:(NSString *)key;

简单示例:

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    NSLog(@"name = %@", [self valueForKey:@"name"]);
    
    id numbers = [self valueForKey:@"numbers"];
    NSLog(@"numbres: %@",NSStringFromClass([numbers class]));
    NSLog(@"numbers = %@", numbers);
    
    [self valueForKey:@"abc"];
}

-(NSString *)getName
{
    return @"getName";
}

-(NSString *)name
{
    return @"name";
}

-(NSString *)isName
{
    return @"isName";
}

- (NSInteger)countOfNumbers
{
    return 5;
}

- (id)objectInNumbersAtIndex:(NSInteger)index
{
    return @(index + 1);
}

- (id)valueForUndefinedKey:(NSString *)key
{
    NSLog(@"valueForUndefinedKey, key = %@", key);
    return nil;
}

@end
结果:
2019-06-22 23:28:58.730209+0800 study[10995:169342] name = getName
2019-06-22 23:28:58.730510+0800 study[10995:169342] numbres: NSKeyValueArray
2019-06-22 23:28:58.730724+0800 study[10995:169342] numbers = (
    1,
    2,
    3,
    4,
    5
)
2019-06-22 23:28:58.730946+0800 study[10995:169342] valueForUndefinedKey, key = abc

上一篇下一篇

猜你喜欢

热点阅读