題外話:
最近在看MJExtension的源碼,整體思路與其他數(shù)據(jù)映射模型一樣。
以字典轉(zhuǎn)模型為例,MJ會(huì)把模型的屬性剝離出來,分別建立一個(gè)對(duì)象去存儲(chǔ)該屬性的類型(例如,該屬性的類,是否為引用類型還是數(shù)據(jù)類型等等),最后利用KVC將字典的值一一賦值到屬性中,從而完成映射。YYModel是采用的setter和getter方法,據(jù)他測(cè)試,速度會(huì)比KVC快。
題內(nèi)話:
什么是KVC,KVC就是鍵值編碼。鍵值編碼看起來有點(diǎn)抽象,你可以理解成一個(gè)對(duì)象就是一個(gè)字典,屬性則為它的鍵,屬性值則為該鍵的值。全名叫NSKeyValueCoding,提供一種機(jī)制來間接訪問對(duì)象的屬性。
KVC常用方法
- (void)setValue:(nullable id)value forKey:(NSString *)key; // 為對(duì)象的屬性賦值
- (id)valueForKey:(NSString *)key; // 根據(jù)key取值
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath; // 為對(duì)象的屬性賦值(包含了setValue:forKey:的功能,并且還可以對(duì)對(duì)象內(nèi)的類的屬性進(jìn)行賦值)
- (nullable id)valueForKeyPath:(NSString *)keyPath; // 根據(jù)keyPath取值
- (void)setValuesForKeysWithDictionary:(NSDictionary<NSString *, id> *)keyedValues; // 對(duì)模型一次性賦值,前提是必須聲明好所有對(duì)應(yīng)的屬性(key)
假設(shè)存在以下模型
#import <Foundation/Foundation.h>
typedef enum {
SexMale,
SexFemale
} Sex;
@interface MJUser : NSObject
/** 名稱 */
@property (copy, nonatomic) NSString *name;
/** 頭像 */
@property (copy, nonatomic) NSString *icon;
/** 年齡 */
@property (assign, nonatomic) unsigned int age;
/** 身高 */
@property (strong, nonatomic) NSNumber *height;
/** 財(cái)富 */
@property (strong, nonatomic) NSDecimalNumber *money;
/** 性別 */
@property (assign, nonatomic) Sex sex;
@end
那么我們可以使用
MJUser *user = [MJUser new];
[user setValue:@"jack" forKey:@"name"];
NSLog(@"userName : %@",[user valueForKey:@"name"]);//jack
這是基本取值賦值
1.我們可以發(fā)現(xiàn)value的值都是id,所以數(shù)據(jù)類型需要裝箱成引用類型。
2.key和keyPath的區(qū)別:keyPath方法是集成了key的所有功能。但是對(duì)對(duì)象中的對(duì)象的屬性進(jìn)行賦值,只有keyPath能夠?qū)崿F(xiàn)
3.當(dāng)key值是沒有定義的,也就是對(duì)象沒有這個(gè)屬性時(shí),如果還調(diào)用setValue:forKey:,那么setValue:forUndefinedKey這個(gè)方法會(huì)被調(diào)用。
#import "MJUser.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
MJUser *user = [MJUser new];
[user setValue:@(100) forKey:@"scroe"];
}
#import "MJUser.h"
@implementation MJUser
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
if ([key isEqualToString:@"scroe"]) {
NSLog(@"設(shè)置了scroe的值為 : %ld",[value integerValue]);//設(shè)置了scroe的值為 : 100
}
@end
底層原理分析
我們看下setValue:forKey的賦值原理
1.去模型中查找有沒有對(duì)應(yīng)的setter方法:例如setIcon方法,有就直接調(diào)用這個(gè)setter方法給這個(gè)屬性賦值[self setIcon:dic[@"icon"]];(PS:這里是不是能看出來為什么YY比MJ快了吧,YY是直接調(diào)用setter和getter方法,不用去尋找)
2.如果找不到setter方法,接著會(huì)去找有沒有icon屬性,如果有,就直接訪問模型中的icon屬性,進(jìn)行賦值,icon=dict[@"icon"];
3.如果找不到icon屬性,接著會(huì)去找_icon屬性,如果有,直接進(jìn)行賦值,_icon = dict[@"icon"];
4.如果都找不到就會(huì)報(bào)錯(cuò),[setValue:forUndefined:]將會(huì)被調(diào)用。
內(nèi)部實(shí)現(xiàn)
[user setValue:@"jack" forKey:@"name"];
就會(huì)被編譯器處理成
SEL sel = sel_get_uid ("setValue:forKey:");
IMP method = objc_msg_lookup (user->isa,sel);
method(user, sel, @"jack", @"name");
我們可知,對(duì)象在調(diào)用setValue的時(shí)候,會(huì)直接獲取這個(gè)方法子,再尋找方法實(shí)現(xiàn)的接口,再直接查找方法的實(shí)現(xiàn).
KVO
KVO,即key-value-observing,利用一個(gè)key來找到某個(gè)屬性并監(jiān)聽它的值的改變。
這也是設(shè)計(jì)模式觀察者模式的體現(xiàn)。
使用非常簡(jiǎn)單
在觀察者實(shí)現(xiàn)監(jiān)聽方法observeValueForKeyPath: ofObject: change: context:
#import "MJUser.h"
@implementation MJUser
- (instancetype)init
{
if (self = [super init]) {
[self addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];
}
return self;
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
if ([keyPath isEqual:@"name"]) {
NSLog(@"newValue: %@",change[NSKeyValueChangeNewKey]);
}
}
MJUser *user = [[MJUser alloc]init];
user.name = @"oldName";//newValue: oldName
user.name = @"newName";//newValue: newName
KVO的底層實(shí)現(xiàn)
當(dāng)一個(gè)類的某個(gè)屬性被監(jiān)聽的時(shí)候,系統(tǒng)會(huì)通過runtime動(dòng)態(tài)的去創(chuàng)建繼承于該類的子類,并重寫了這個(gè)子類屬性的setter方法,同時(shí),會(huì)將父類的isa指針指向子類,從而實(shí)現(xiàn)調(diào)用子類被重寫的setter方法。重寫的setter方法會(huì)在調(diào)用原setter方法前后,通知觀察對(duì)象屬性值的改變
//被重寫的setter方法
- (void)setName:(NSString *)name
{
[self willChangeValueForKey:@"name"];
[super setValue:name forKey:@"name"];
[self didChangeValueForKey:@"name"];
}
那么我們還是不知道怎么通知到對(duì)象屬性值的改變。
其實(shí)就是在didChangeValueForKey:方法內(nèi)部調(diào)用的。
我們可以手寫一個(gè)KVO驗(yàn)證一下
@interface ViewController ()
@property (nonatomic, assign) BOOL new;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self addObserver:self forKeyPath:@"new" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"1");
[self willChangeValueForKey:@"new"];
NSLog(@"2");
[self didChangeValueForKey:@"new"];
NSLog(@"4");
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
NSLog(@"3");
}
//
2018-06-07 19:31:11.140346+0800 KVOTest[30805:1532911] 1
2018-06-07 19:31:11.140498+0800 KVOTest[30805:1532911] 2
2018-06-07 19:31:11.140620+0800 KVOTest[30805:1532911] 3
2018-06-07 19:31:11.140714+0800 KVOTest[30805:1532911] 4
假如我們注釋掉didChangeValueForKey:
結(jié)果會(huì)是怎樣的
@interface ViewController ()
@property (nonatomic, assign) BOOL new;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self addObserver:self forKeyPath:@"new" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"1");
[self willChangeValueForKey:@"new"];
NSLog(@"2");
// [self didChangeValueForKey:@"new"];
NSLog(@"4");
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
NSLog(@"3");
}
2018-06-07 19:32:11.295735+0800 KVOTest[30833:1536910] 1
2018-06-07 19:32:11.295883+0800 KVOTest[30833:1536910] 2
2018-06-07 19:32:11.295964+0800 KVOTest[30833:1536910] 4
結(jié)果一看而知