KVO/KVC

什么是KVO

  • KVO是Key-Value Observing的首字母縮寫(xiě)
  • KVO是Object-C對(duì)觀察者設(shè)計(jì)模式的實(shí)現(xiàn)
  • Apple使用了isa混寫(xiě)(isa-swizzling)來(lái)實(shí)現(xiàn)KVO

KVO 提供一種機(jī)制浇辜,指定一個(gè)被觀察對(duì)象(例如 A 類(lèi))碧绞,當(dāng)對(duì)象某個(gè)屬性(例如 A 中的字符串 name)發(fā)生更改時(shí)豁翎,對(duì)象會(huì)獲得通知漓摩,并作出相應(yīng)處理;【且不需要給被觀察的對(duì)象添加任何額外代碼膊爪,就能使用 KVO 機(jī)制】
用一張圖來(lái)描述一下KVO的實(shí)現(xiàn)機(jī)制

KVO

上圖可以看出,注冊(cè)一個(gè)對(duì)象的觀察者的時(shí)候,實(shí)際上是調(diào)用了系統(tǒng)的- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;這個(gè)方法,調(diào)用這個(gè)方法后觀察者觀察對(duì)象A中的某個(gè)屬性,然后系統(tǒng)會(huì)在運(yùn)行時(shí)動(dòng)態(tài)的創(chuàng)建一個(gè)NSKVONotifying_A的這么樣一個(gè)類(lèi),原來(lái)的對(duì)象A的isa指針重新指向了NSKVONotifying_A這個(gè)類(lèi),把isa的指向進(jìn)行修改就是isa混寫(xiě)技術(shù).NSKVONotifying_A是類(lèi)A的子類(lèi),并重寫(xiě)了其中的Setter方法,通過(guò)對(duì)Setter方法的重寫(xiě)達(dá)到可以通知所有觀察者的目的.
接下來(lái),在XCode工程當(dāng)中,來(lái)實(shí)際通過(guò)Setter方法的設(shè)置,KVO的監(jiān)聽(tīng)來(lái)感受一下KVO的實(shí)現(xiàn).
創(chuàng)建兩個(gè)文件,MyObject和MyObserver.

  • MyObject
@interface MyObject : NSObject
@property (nonatomic,assign) int value;
-(void)increase;
@end
@implementation MyObject

-(instancetype)init
{
    self = [super init];
    if (self) {
        _value = 0;
    }
    return self;
}

-(void)increase
{
    _value += 1;
}

@end
  • MyObserver
@implementation MyObserver
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
    NSNumber *valueNum = [change valueForKey:NSKeyValueChangeNewKey];
    NSLog(@"value is %@",valueNum);
}
@end

然后在AppDelegate中進(jìn)行KVO的監(jiān)聽(tīng)

  • AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    
    MyObject *obj = [[MyObject alloc]init];
    MyObserver *observer = [[MyObserver alloc]init];
    
    //調(diào)用KVO方法監(jiān)聽(tīng)obj的value屬性的變化
    [obj addObserver:observer forKeyPath:@"value" options:NSKeyValueObservingOptionNew context:NULL];
    obj.value = 1;
    return YES;
}

可以看到控制臺(tái)打印出了結(jié)果


說(shuō)明監(jiān)聽(tīng)成功了.
在obj.value那里打個(gè)斷點(diǎn),看看MyObject是怎么被改寫(xiě)的.


不出所料,在監(jiān)聽(tīng)的屬性Value被改寫(xiě)后,MyObject變成了NSKVONotifying_MyObject了.
為什么調(diào)用Setter方法就可以實(shí)現(xiàn)這種KVO的監(jiān)聽(tīng)呢.

重寫(xiě)的Setter添加的方法
  • -(void)willChangeValueForKey:(NSString *)key
  • -(void)didChangeValueForKey:(NSString *)key

那么在NSKVONotifying_MyObject中的Setter方法就變成了下面這樣

-(void)setValue:(id)obj
{
    [self willChangeValueForKey:@"keyPath"];
    //調(diào)用父類(lèi),也就是原類(lèi)的實(shí)現(xiàn)
    [super setValue:obj];
    [self didChangeValueForKey:@"keyPath"];
}

接下來(lái)有兩個(gè)問(wèn)題.

1.通過(guò)KVC設(shè)置Value能否生效

這個(gè)問(wèn)題用代碼來(lái)驗(yàn)證一下就可以了,如下所示

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    MyObject *obj = [[MyObject alloc]init];
    MyObserver *observer = [[MyObserver alloc]init];
    //調(diào)用KVO方法監(jiān)聽(tīng)obj的value屬性的變化
    [obj addObserver:observer forKeyPath:@"value" options:NSKeyValueObservingOptionNew context:NULL];
    obj.value = 1;
    // 使用kvc來(lái)改變value的值
    [obj setValue:@2 forKey:@"value"];
    return YES;
}

結(jié)果控制臺(tái)打印如下:



說(shuō)明使用KVC設(shè)置屬性的方式是可以出發(fā)KVO的,說(shuō)明KVC設(shè)置屬性是觸發(fā)了Setter方法

2.使用成員變量賦值會(huì)出發(fā)KVO嗎

我們?cè)贏ppDelegate調(diào)用obj的increase方法,發(fā)現(xiàn)控制臺(tái)只打印了value is 1,說(shuō)明對(duì)成員變量賦值不會(huì)觸發(fā)KVO,但對(duì)increate方法進(jìn)行以下操作就不一樣了.
不過(guò)如果我們把increase方法變成下面這樣,再運(yùn)行試試

-(void)increase
{
    [self willChangeValueForKey:@"value"];
    _value += 1;
    [self didChangeValueForKey:@"value"];
}

發(fā)現(xiàn)又觸發(fā)了KVO
所以根據(jù)上面的實(shí)驗(yàn)總結(jié)出下面幾點(diǎn)

  • 使用setter方法改變值KVO才會(huì)生效
  • 使用setValue:forKey:改變KVO才會(huì)生效
  • 成員變量直接修改需手動(dòng)添加KVO才會(huì)生效

KVC

KVC是Key-Value coding的縮寫(xiě),也就是鍵值編碼,和鍵值編碼相關(guān)的兩個(gè)方法就是下面這兩個(gè)

  • -(id)valueForKey:(NSString *)key
    這個(gè)可以調(diào)用某個(gè)實(shí)例的ValueForKey:方法,來(lái)獲取和key同名或相似名稱(chēng)的實(shí)例變量的值
  • -(void)setValue forKey:(NSString *)key
    根據(jù)這個(gè)方法可以設(shè)置某一個(gè)對(duì)象和這個(gè)key同名或者相似名稱(chēng)的實(shí)例變量的值.
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末自阱,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子米酬,更是在濱河造成了極大的恐慌沛豌,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,695評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件赃额,死亡現(xiàn)場(chǎng)離奇詭異加派,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)跳芳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)芍锦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人飞盆,你說(shuō)我怎么就攤上這事娄琉。” “怎么了吓歇?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,130評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵孽水,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我城看,道長(zhǎng)匈棘,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,648評(píng)論 1 297
  • 正文 為了忘掉前任析命,我火速辦了婚禮,結(jié)果婚禮上逃默,老公的妹妹穿的比我還像新娘鹃愤。我一直安慰自己,他們只是感情好完域,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布软吐。 她就那樣靜靜地躺著,像睡著了一般吟税。 火紅的嫁衣襯著肌膚如雪凹耙。 梳的紋絲不亂的頭發(fā)上姿现,一...
    開(kāi)封第一講書(shū)人閱讀 52,268評(píng)論 1 309
  • 那天,我揣著相機(jī)與錄音肖抱,去河邊找鬼备典。 笑死,一個(gè)胖子當(dāng)著我的面吹牛意述,可吹牛的內(nèi)容都是我干的提佣。 我是一名探鬼主播,決...
    沈念sama閱讀 40,835評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼荤崇,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼拌屏!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起术荤,我...
    開(kāi)封第一講書(shū)人閱讀 39,740評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤倚喂,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后瓣戚,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體端圈,經(jīng)...
    沈念sama閱讀 46,286評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評(píng)論 3 340
  • 正文 我和宋清朗相戀三年带兜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了枫笛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,505評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡刚照,死狀恐怖刑巧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情无畔,我是刑警寧澤啊楚,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站浑彰,受9級(jí)特大地震影響恭理,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜郭变,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評(píng)論 3 333
  • 文/蒙蒙 一颜价、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧诉濒,春花似錦周伦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,357評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春寨腔,著一層夾襖步出監(jiān)牢的瞬間速侈,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,466評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工迫卢, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留倚搬,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,921評(píng)論 3 376
  • 正文 我出身青樓靖避,卻偏偏與公主長(zhǎng)得像潭枣,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子幻捏,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評(píng)論 2 359

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

  • 1.KVO KVO即鍵值監(jiān)聽(tīng)。KVO模式在廣泛應(yīng)用的MVC模式中應(yīng)用很廣泛榛臼。在C中注冊(cè)C為M中屬性的監(jiān)聽(tīng)者伊佃,當(dāng)M中...
    雪山飛狐_91ae閱讀 585評(píng)論 0 1
  • 在iOS開(kāi)發(fā)過(guò)程中,我們經(jīng)常會(huì)聽(tīng)到或者用到KVO沛善,KVC航揉,NSNotificationCenter等,但是很多時(shí)候...
    dullgrass閱讀 7,078評(píng)論 14 133
  • KVC KVC定義 KVC(Key-value coding)鍵值編碼金刁,就是指iOS的開(kāi)發(fā)中帅涂,可以允許開(kāi)發(fā)者通過(guò)K...
    暮年古稀ZC閱讀 2,148評(píng)論 2 9
  • KVO: KVO的全稱(chēng)是Key-Value Observing,俗稱(chēng)“鍵值監(jiān)聽(tīng)”尤蛮,可以用于監(jiān)聽(tīng)某個(gè)對(duì)象屬性值的改變...
    曹來(lái)東閱讀 631評(píng)論 0 0
  • Key Value Observing(KVO)鍵值觀察产捞,它提供一種機(jī)制醇锚,當(dāng)指定的對(duì)象的屬性被修改后,則對(duì)象就會(huì)接...
    小小土豆dev閱讀 232評(píng)論 0 5