iOS - 自定義KVO

之前我們已經(jīng)了解過了KVO的底層實現(xiàn)原理,不過呢,在我們開始實現(xiàn)自定義KVO之前再來簡單回顧下KVO的實現(xiàn)原理

1.創(chuàng)建子類
2.重寫一個setter方法(其實是添加一個setter方法)
3.修改isa指針指向新創(chuàng)建的子類
4.調(diào)用父類的setName方法
5.將觀察者保存到當(dāng)前對象

接下來我們開始自定義KVO

我們先創(chuàng)建一個NSObject分類;然后實現(xiàn)對應(yīng)的方法

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface NSObject (KVO)

- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;

@end

NS_ASSUME_NONNULL_END

1.創(chuàng)建子類

在.m文件里#import <objc/message.h>
創(chuàng)建完子類記得要注冊這個類,

- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
    
    //1.創(chuàng)建子類
    NSString *oldClassName = NSStringFromClass(self.class);
    NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
    
    Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
    
    // 注冊
    objc_registerClassPair(myClass);    
    
}

2.重寫一個setter方法(其實是添加一個setter方法)

- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
    
    //1.創(chuàng)建子類
    NSString *oldClassName = NSStringFromClass(self.class);
    NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
    
    Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
    
    // 注冊
    objc_registerClassPair(myClass);

    //2.重新setNanme方法(其實是添加一個setName方法)
    class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");
}

void setMethod(id self,SEL _cmd,NSString *newName){
   
    NSLog(@"來了%@",newName);
}

3.修改isa指針指向新創(chuàng)建的子類

- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
    
    //1.創(chuàng)建子類  
    NSString *oldClassName = NSStringFromClass(self.class);
    NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
    
    Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
    
    // 注冊
    objc_registerClassPair(myClass);

    //2.重新setNanme方法(其實是添加一個setName方法)
    class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");

    //3.修改imp地址
    object_setClass(self, myClass);
}

這個時候我們在控制器中注冊這個觀察者( 其中person是需要觀察屬性變化的實體類)

- (void)viewDidLoad {
    [super viewDidLoad];

    person = [[Person alloc]init];
    [person ZXY_addObserver:self forKeyPath:@"name" options:(NSKeyValueObservingOptionNew) context:nil];

}
//點(diǎn)擊屏幕
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    
    static int a;
    person.name = [NSString stringWithFormat:@"%d",a++];
}

然后我們跑起來,點(diǎn)擊屏幕: 控制臺打印


屏幕快照 2018-12-25 下午4.33.43.png

4.調(diào)用父類的setName方法

到目前這個階段,我們已經(jīng)能獲取到修改的值了,接下來我們需要調(diào)用父類的set方法,去修改屬性值,這樣person的name就修改了.

void setName(id self,SEL _cmd,NSString *newName){
    
    NSLog(@"來了%@",newName);
    
    //5. 調(diào)用父類的setName方法
    Class class = [self class];//拿到當(dāng)前類型

    object_setClass(self, class_getSuperclass(class));// 將當(dāng)前類型設(shè)置為父類類型

    objc_msgSend(self, @selector(setName:),newName);//給父類的set方法發(fā)送消息,傳遞修改的值

    // 改為子類
    object_setClass(self, class);
}
注意:在這一步一定要將當(dāng)前的類型設(shè)置為父類類型,設(shè)置完后也一定要改為子類的類型

5.將觀察者保存到當(dāng)前對象

現(xiàn)在我們需要通知外部方法,已經(jīng)將屬性修改了,并將之傳遞出去,所以我們設(shè)置一個觀察者

- (void)ZXY_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context{
    
    //1.創(chuàng)建子類   
    NSString *oldClassName = NSStringFromClass(self.class);
    NSString *newClassName = [@"ZXYKVO_" stringByAppendingString:oldClassName];
    
    Class myClass = objc_allocateClassPair(self.class, newClassName.UTF8String, 0);
    
    // 注冊
    objc_registerClassPair(myClass);

    //2.重新setNanme方法(其實是添加一個setName方法)
    /**參數(shù)
     *class 給哪個類添加方法
     *sel   方法編號
     *imp   方法實現(xiàn)(函數(shù)指針)
     *type  返回值類型 ()
     */
    class_addMethod(myClass, @selector(setName:), (IMP)setMethod, "v@:@");

    //3.修改imp地址
    object_setClass(self, myClass);

    //4.將觀察者保存到當(dāng)前對象
   /**
     id object                     :表示關(guān)聯(lián)者鲸郊,是一個對象,變量名理所當(dāng)然也是object
     const void *key               :獲取被關(guān)聯(lián)者的索引key
     id value                      :被關(guān)聯(lián)者,這里是一個block
     objc_AssociationPolicy policy : 關(guān)聯(lián)時采用的協(xié)議,有assign,retain缴守,copy等協(xié)議,一般使用OBJC_ASSOCIATION_RETAIN_NONATOMIC,這里我們使用Weak協(xié)議,避免循環(huán)引用
     */
    objc_setAssociatedObject(self, "observer", observer, OBJC_ASSOCIATION_ASSIGN);//
}
在這里就有可能就有人問了,明明set方法只有一個返回類型,為什么要寫“v@:@”,這里解釋下,V表示的是返回void類型,第二個@表示的是一個id對象,第三個:表示的是方法SEL,最后一個@表示的傳入的變量值.
??注意:以為OC方法會默認(rèn)兩個參數(shù),一個id對象,一個方法SEL

然后我們在set方法中獲取到這個觀察者

void setName(id self,SEL _cmd,NSString *newName){

    NSLog(@"來了%@",newName);
    
    //5. 調(diào)用父類的setName方法
    Class class = [self class];//拿到當(dāng)前類型

    object_setClass(self, class_getSuperclass(class));

    objc_msgSend(self, @selector(setName:),newName);

    // 觀察者
    id observer = objc_getAssociatedObject(self, "observer");

    // 改為子類
    object_setClass(self, class);
}

給系統(tǒng)的observeValueForKeyPath方法發(fā)送消息

void setName(id self,SEL _cmd,NSString *newName){

    NSLog(@"來了%@",newName);
    
    //5. 調(diào)用父類的setName方法
    Class class = [self class];//拿到當(dāng)前類型

    object_setClass(self, class_getSuperclass(class));

    objc_msgSend(self, @selector(setName:),newName);

    // 觀察者
    id observer = objc_getAssociatedObject(self, "observer");

    if (observer) {
        objc_msgSend(observer, @selector(observeValueForKeyPath:ofObject:change:context:),@"name",self,@{@"new:":newName,@"kind:":@1},nil);
    }

    // 改為子類
    object_setClass(self, class);
}

其中observeValueForKeyPath方法的參數(shù)說明:
keyPath: 屬性名字
object: 哪個對象
change: 一個字典,它描述對鍵路徑keyPath中的屬性值相對于對象所做的更改
context: 當(dāng)注冊觀察者以接收鍵值觀察通知時提供的值,這里寫為nil就行.

最后

現(xiàn)在我們在外部控制器中打印修改后的值

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
    
    NSLog(@"%@",change);
    
}
屏幕快照 2018-12-26 下午2.23.06.png

到這里,自定義KVO就簡單實現(xiàn)了,但是目前里面的屬性變量名都是固定,如果能自定義的變化,那就更完美了.降下來,我們就來慢慢的研究他.

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市焚廊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌习劫,老刑警劉巖咆瘟,帶你破解...
    沈念sama閱讀 219,188評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異诽里,居然都是意外死亡袒餐,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評論 3 395
  • 文/潘曉璐 我一進(jìn)店門须肆,熙熙樓的掌柜王于貴愁眉苦臉地迎上來匿乃,“玉大人,你說我怎么就攤上這事豌汇〈闭ǎ” “怎么了?”我有些...
    開封第一講書人閱讀 165,562評論 0 356
  • 文/不壞的土叔 我叫張陵拒贱,是天一觀的道長宛徊。 經(jīng)常有香客問我,道長逻澳,這世上最難降的妖魔是什么闸天? 我笑而不...
    開封第一講書人閱讀 58,893評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮斜做,結(jié)果婚禮上苞氮,老公的妹妹穿的比我還像新娘。我一直安慰自己瓤逼,他們只是感情好笼吟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,917評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著霸旗,像睡著了一般贷帮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上诱告,一...
    開封第一講書人閱讀 51,708評論 1 305
  • 那天撵枢,我揣著相機(jī)與錄音,去河邊找鬼。 笑死锄禽,一個胖子當(dāng)著我的面吹牛潜必,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播沟绪,決...
    沈念sama閱讀 40,430評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼刮便,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了绽慈?” 一聲冷哼從身側(cè)響起恨旱,我...
    開封第一講書人閱讀 39,342評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎坝疼,沒想到半個月后搜贤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,801評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡钝凶,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,976評論 3 337
  • 正文 我和宋清朗相戀三年仪芒,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片耕陷。...
    茶點(diǎn)故事閱讀 40,115評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡掂名,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出哟沫,到底是詐尸還是另有隱情饺蔑,我是刑警寧澤,帶...
    沈念sama閱讀 35,804評論 5 346
  • 正文 年R本政府宣布嗜诀,位于F島的核電站猾警,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏隆敢。R本人自食惡果不足惜发皿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,458評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拂蝎。 院中可真熱鬧穴墅,春花似錦、人聲如沸温自。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽捣作。三九已至誉结,卻和暖如春鹅士,著一層夾襖步出監(jiān)牢的瞬間券躁,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留也拜,地道東北人以舒。 一個月前我還...
    沈念sama閱讀 48,365評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像邻眷,于是被迫代替她去往敵國和親吗蚌。 傳聞我的和親對象是個殘疾皇子稽屏,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,055評論 2 355

推薦閱讀更多精彩內(nèi)容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴(yán)謹(jǐn) 對...
    cosWriter閱讀 11,103評論 1 32
  • 利用Runtime 實現(xiàn)簡單的自定義kvo 代碼githubgithub.com/zswj/custom-KVO ...
    我是數(shù)據(jù)鏈路層閱讀 855評論 1 0
  • 自己實現(xiàn)kvo之前,需要知道iOS系統(tǒng)對kvo的實現(xiàn)滥沫。 系統(tǒng)實現(xiàn)kvo的原理 這依賴了OC強(qiáng)大的runtime特性...
    mws100閱讀 2,779評論 6 3
  • 當(dāng)某個類的屬性對象第一次被觀察時,系統(tǒng)就會在運(yùn)行期動態(tài)地創(chuàng)建該類的一個派生類键俱,在這個派生類中重寫基類中任何被觀察屬...
    linbj閱讀 967評論 0 13
  • “你一定要守護(hù)我們的村民编振,還有你給我做的星空”缀辩。這是電影里的橋段,卻讓我感受到真實踪央。阿月與天蓬不是主角臀玄,戲份也不多...
    mr莫冉閱讀 561評論 0 1