從MJExtension引發(fā)的對(duì)KVC和KVO的見解

題外話:
最近在看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é)果一看而知

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • KVC KVC定義 KVC(Key-value coding)鍵值編碼,就是指iOS的開發(fā)中,可以允許開發(fā)者通過K...
    暮年古稀ZC閱讀 2,291評(píng)論 2 9
  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時(shí)...
    歐辰_OSR閱讀 30,232評(píng)論 8 265
  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴(yán)謹(jǐn) 對(duì)...
    cosWriter閱讀 11,652評(píng)論 1 32
  • 該文章屬于劉小壯原創(chuàng),轉(zhuǎn)載請(qǐng)注明:劉小壯[http://www.itdecent.cn/u/2de707c93d...
    劉小壯閱讀 21,503評(píng)論 29 115
  • KVC(Key-valuecoding)鍵值編碼,單看這個(gè)名字可能不太好理解。其實(shí)翻譯一下就很簡(jiǎn)單了,就是指iOS...
    榕樹頭閱讀 770評(píng)論 0 2

友情鏈接更多精彩內(nèi)容