iOS KVC(一)基本了解
iOS KVC (二) 不可不知的賦值深層次原理
iOS KVC (三)不可不知的取值深層次原理
iOS KVC (四)keyPath的深度解析
iOS KVC (五)KVC幾種典型的異常處理
iOS KVC (六) KVC容器類及深層次原理
iOS KVC(七) KVC正確性的驗(yàn)證
iOS KVC (八) KVC幾種常見(jiàn)應(yīng)用
iOS KVC (九) KVC模型轉(zhuǎn)化(1) 模型打印 description, debugDescription
iOS KVC (十)模型轉(zhuǎn)換(2)模型轉(zhuǎn)換
KVC賦值
賦值的過(guò)程看似很簡(jiǎn)單,就是找到對(duì)應(yīng)的key,然后將一個(gè)已經(jīng)準(zhǔn)備好的值賦過(guò)去,但是你知道深層次的原理嗎?方法- (nullable id)valueForKey:(NSString *)key;中的key系統(tǒng)內(nèi)部和底層是按照什么邏輯找的呢?下面就讓我們看一下。
下面就是尋找key的內(nèi)部流程。以[self setValue:@"小明" forKey:@"name"];這句代碼作為例子進(jìn)行說(shuō)明。
程序優(yōu)先調(diào)用set<Key>:屬性值方法,代碼通過(guò)setter方法完成設(shè)置。注意,這里的<key>是指成員變量名,首字母大小寫要符合KVC的命名規(guī)則,下同。
如果沒(méi)有找到setName:方法,KVC機(jī)制會(huì)檢查+ (BOOL)accessInstanceVariablesDirectly方法有沒(méi)有返回YES,默認(rèn)該方法會(huì)返回YES,如果你重寫了該方法讓其返回NO的話,那么在這一步KVC會(huì)執(zhí)行setValue:forUndefinedKey:方法,不過(guò)一般開(kāi)發(fā)者不會(huì)這么做。所以KVC機(jī)制會(huì)搜索該類里面有沒(méi)有名為<key>的成員變量,無(wú)論該變量是在類接口處定義,還是在類實(shí)現(xiàn)處定義,也無(wú)論用了什么樣的訪問(wèn)修飾符,只在存在以<key>命名的變量,KVC都可以對(duì)該成員變量賦值。
如果該類即沒(méi)有set<key>:方法,也沒(méi)有_<key>成員變量,KVC機(jī)制會(huì)搜索_is<Key>的成員變量。
和上面一樣,如果該類即沒(méi)有set<Key>:方法,也沒(méi)有_<key>和_is<Key>成員變量,KVC機(jī)制再會(huì)繼續(xù)搜索<key>和is<Key>的成員變量。再給它們賦值。
如果上面列出的方法或者成員變量都不存在,系統(tǒng)將會(huì)執(zhí)行該對(duì)象的setValue:forUndefinedKey:方法,默認(rèn)是拋出異常。
特別需要注意的是:如果開(kāi)發(fā)者想讓這個(gè)類禁用KVC里,那么重寫+ (BOOL)accessInstanceVariablesDirectly方法讓其返回NO即可,這樣的話如果KVC沒(méi)有找到set<Key>:屬性名時(shí),會(huì)直接用setValue:forUndefinedKey:方法。
下面結(jié)合例子進(jìn)行說(shuō)明
+ (BOOL)accessInstanceVariablesDirectly 返回NO實(shí)例
實(shí)例
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
{
NSString *name;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self setValue:@"小明" forKey:@"name"];
NSLog(@"name = %@", name);
}
+ (BOOL)accessInstanceVariablesDirectly
{
return NO;
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
NSLog(@"沒(méi)有找到該key對(duì)應(yīng)的屬性會(huì)拋出異常");
}
打印數(shù)據(jù):
2018-05-16 14:45:42.959735+0700 KVC[21922:462925] 沒(méi)有找到該key對(duì)應(yīng)的屬性會(huì)拋出異常
2018-05-16 14:45:42.959850+0700 KVC[21922:462925] name = (null)
總結(jié):這里+ (BOOL)accessInstanceVariablesDirectly返回了NO,也就是說(shuō)找不到屬性的setter方法,那么不會(huì)再去找實(shí)例變量,所以會(huì)輸出上面的結(jié)果。所以name輸出結(jié)果就是null。
+ (BOOL)accessInstanceVariablesDirectly 返回YES實(shí)例
accessInstanceVariablesDirectly 默認(rèn)返回YES。
實(shí)例1 self.name
#import "ViewController.h"
@interface ViewController ()
@property (nonatomic,strong)NSString *name;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self setValue:@"小明" forKey:@"name"];
NSLog(@"self.name = %@", self.name);
}
實(shí)例2 name
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
{
NSString *name;
}
- (void)viewDidLoad {
[super viewDidLoad];
[self setValue:@"小明" forKey:@"name"];
NSLog(@"name = %@", name);
}
打印數(shù)據(jù):
2018-05-16 14:53:45.648667+0700 KVC[22042:467903] name = 小明
實(shí)例3 _name
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
{
NSString *_name;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self setValue:@"小明" forKey:@"name"];
NSLog(@"_name = %@", _name);
}
打印數(shù)據(jù):
2018-05-16 14:57:03.714227+0700 KVC[22107:470305] _name = 小明
實(shí)例4 _isName
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
{
NSString *_isName;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self setValue:@"小明" forKey:@"name"];
NSLog(@"_isName = %@", _isName);
}
打印數(shù)據(jù):
2018-05-16 14:58:24.632017+0700 KVC[22153:471747] _isName = 小明
實(shí)例5 isName
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
{
NSString *isName;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self setValue:@"小明" forKey:@"name"];
NSLog(@"isName = %@", isName);
}
打印數(shù)據(jù):
2018-05-16 15:00:53.388740+0700 KVC[22223:474376] isName = 小明
下面就是我們這個(gè)屬性和實(shí)例變量的調(diào)用順序。
- self.name
- _name
- _isName
- name
- isName
上面的順序親測(cè),是按照所列順序執(zhí)行的。