KVO 原理探究

[TOC]

KVO

研究

沒有使用KVO和使用KVO的變化

測(cè)試的類Person

@interface Person : NSObject
@property (nonatomic, assign) int age;
@end

@implementation Person
@end

通過 objc_copyClassList 驗(yàn)證

思路: 使用runmtimeobjc_copyClassList 獲取所有的類浅侨,對(duì)比前后的變化


未使用KVO

    unsigned int count = 0;
    Class *classes = objc_copyClassList(&count);
    for (int i = 0; i < count; i++) {
        Class class = classes[i];
        const char *name = class_getName(class);
        NSLog(@"classname: %s", name);
    }

輸出:

classname: Person

實(shí)際上會(huì)打印很多類盔腔,上面只截取了相關(guān)的類


使用了KVO (注意:這里需要先添加監(jiān)聽竹椒,然后再使用objc_copyClassList方法獲取岛啸,否則是獲取不到的,因?yàn)?code>KVO是動(dòng)態(tài)添加的)

    Person *obj = [[Person alloc] init];
    [obj addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
    
    unsigned int count = 0;
    Class *classes = objc_copyClassList(&count);
    for (int i = 0; i < count; i++) {
        Class class = classes[i];
        const char *name = class_getName(class);
        NSLog(@"classname: %s", name);
    }

輸出:

classname: Person
classname: NSKVONotifying_Person

可以看到多了一個(gè)類 NSKVONotifying_Person


通過 isa 來驗(yàn)證

image
  • 添加 kvo 前后 isa 指向:
    • 之前: Person
    • 之后: NSKVONotifying_Person
  • 添加 kvo 前后 po 命令輸出的都是 Person

小結(jié):
1掌动、使用KVO讹堤,runtime會(huì)給需要監(jiān)聽的類 Person 自動(dòng)創(chuàng)建一個(gè)類 NSKVONotifying_Person
2、并且把原來類創(chuàng)建的對(duì)象的 isa 指向了新建的類 NSKVONotifying_Person


通過 objc 源碼可以看到 Class changeIsa(Class newCls); 這個(gè)函數(shù)來改變 isa


自動(dòng)創(chuàng)建的類 NSKVONotifying_Person 的一些特性

使用objc_getClass直接獲取類

父類/元類

測(cè)試代碼如下:

     self.person = [[Person alloc] init];
    
    NSLog(@"添加KVO之前 ======");
    
    {
        Class objClass = object_getClass(self.person);
        Class metaClass = objc_getMetaClass(class_getName(objClass));
        
        NSLog(@"objClass: %@", objClass);
        NSLog(@"metaclass: %@", metaClass);
        
        Class superClass = class_getSuperclass(objClass);
        Class superMetaClass = class_getSuperclass(metaClass);
        
        NSLog(@"superClass: %@", superClass);
        NSLog(@"superMetaClass: %@", superMetaClass);
    }
    
    NSKeyValueObservingOptions options =  NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew;
    [self.person addObserver:self
                  forKeyPath:@"age"
                     options:options
                     context:nil];
    NSLog(@"添加KVO之后 ======");
    
    {
        Class objClass = object_getClass(self.person);
        Class metaClass = objc_getMetaClass(class_getName(objClass));
        
        NSLog(@"objClass: %@", objClass);
        NSLog(@"metaclass: %@", metaClass);
        
        Class superClass = class_getSuperclass(objClass);
        Class superMetaClass = class_getSuperclass(metaClass);
        
        NSLog(@"superClass: %@", superClass);
        NSLog(@"superMetaClass: %@", superMetaClass);
    }

輸出結(jié)果如下:

添加KVO之前 ======
objClass: Person
metaclass: Person
superClass: NSObject
superMetaClass: NSObject
添加KVO之后 ======
objClass: NSKVONotifying_Person
metaclass: NSKVONotifying_Person
superClass: Person
superMetaClass: Person

小結(jié):
因此KVO就是:
1牌捷、通過運(yùn)行時(shí) runtime 新建了一個(gè) 繼承于原來類(Person)的子類(NSKVONotifying_Person)
2墙牌、并且把原來類創(chuàng)建的對(duì)象的isa指向了新建的類 NSKVONotifying_Person

成員變量

    Class kvoClass = objc_getClass("NSKVONotifying_Person");
    
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList(kvoClass, &count);
    NSLog(@"count: %@", @(count));
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        const char *name = ivar_getName(ivar);
        ptrdiff_t offset = ivar_getOffset(ivar);
        const char *type = ivar_getTypeEncoding(ivar);
        NSLog(@"======");
        NSLog(@"name: %s", name);
        NSLog(@"offset: %td", offset);
        NSLog(@"type: %s", type);
    }

輸出如下:

count: 0

小結(jié):
KVO 生成的子類 NSKVONotifying_Person 沒有額外的成員變量

屬性

    Class kvoClass = objc_getClass("NSKVONotifying_Person");
    
    unsigned int propertyCount = 0;
    objc_property_t *properties = class_copyPropertyList(kvoClass, &propertyCount);
    NSLog(@"propertyCount: %d", propertyCount);
    for (int j = 0; j < propertyCount ; j++) {
        objc_property_t property = properties[j];
        const char *name = property_getName(property);
        NSLog(@"propertyName: %s", name);
        const char *attributes = property_getAttributes(property);
        NSLog(@"propertyAttributes: %s", attributes);
    }

輸出:

propertyCount: 0

小結(jié):
KVO 生成的子類 NSKVONotifying_Person,沒有添加額外的屬性

實(shí)例方法

    Class kvoClass = objc_getClass("NSKVONotifying_Person");
    
    unsigned int methodCount = 0;
    Method *methods = class_copyMethodList(kvoClass, &methodCount);
    NSLog(@"methodCount: %d", methodCount);
    for (int k = 0; k < methodCount; k++) {
        Method method = methods[k];
        const char *name = sel_getName(method_getName(method));
        const char *typeEncoding = method_getTypeEncoding(method);
        NSLog(@"分割線 --------------");
        NSLog(@"methodName: %s", name);
        NSLog(@"methodTypeEncoding: %s", typeEncoding);
    }

輸出:

methodCount: 4
分割線 --------------
methodName: setAge:
methodTypeEncoding: v20@0:8i16
分割線 --------------
methodName: class
methodTypeEncoding: #16@0:8
分割線 --------------
methodName: dealloc
methodTypeEncoding: v16@0:8
分割線 --------------
methodName: _isKVOA
methodTypeEncoding: B16@0:8

小結(jié):
1暗甥、KVO 生成的子類 NSKVONotifying_Person 多了四個(gè)實(shí)例方法

  • setAge:: 參數(shù)為 int喜滨,返回值為 void,父類Person也有這個(gè)方法淋袖,子類重寫
  • class: 沒有參數(shù)鸿市,返回值為 #锯梁,表示 class即碗,也就是重寫了 class 方法
  • _isKVOA: 沒有參數(shù),返回值類型為 BOOL
  • dealloc: 重寫了 dealloc 方法

class_isKVOA 方法
    Class kvoClass = objc_getClass("NSKVONotifying_Person");
    id kvoobj = [[kvoClass alloc] init];
    
    unsigned int methodCount = 0;
    Method *methods = class_copyMethodList(kvoClass, &methodCount);
    NSLog(@"methodCount: %d", methodCount);
    for (int k = 0; k < methodCount; k++) {
        Method method = methods[k];
        const char *name = sel_getName(method_getName(method));
        const char *typeEncoding = method_getTypeEncoding(method);
        NSLog(@"分割線 --------------");
        NSLog(@"methodName: %s", name);
        NSLog(@"methodTypeEncoding: %s", typeEncoding);
        
        if ([[NSString stringWithCString:name encoding:NSUTF8StringEncoding] isEqualToString:@"_isKVOA"]) {
            
            bool result = ((bool (*)(id, SEL))(void *)objc_msgSend)((id)kvoobj, method_getName(method));
            NSLog(@"_isKVOA: %@", @(result));
        }
        
        if ([kvoobj respondsToSelector:@selector(class)]) {
            
            id resut = [kvoobj class];
            NSLog(@"class: %@", resut);
        }
    }

輸出結(jié)果為:

  • _isKVOA: 1: 表示返回值是YES
  • class: Person: 還是指向了原來的類陌凳,不需要公開
重寫的 setter 方法
self.person = [[Person alloc] init];
    
    NSLog(@"添加KVO之前 ======");
    
    {
        IMP imp = [self.person methodForSelector:@selector(setAge:)];
        NSLog(@"%p", imp);
    }
    
    NSKeyValueObservingOptions options =  NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew;
    [self.person addObserver:self
                  forKeyPath:@"age"
                     options:options
                     context:nil];
    
    NSLog(@"添加KVO之后 ======");
    
    {
        IMP imp = [self.person methodForSelector:@selector(setAge:)];
        NSLog(@"%p", imp);
    }

結(jié)果如下:

image
  • 使用 KVO 之后 setter 方法的實(shí)現(xiàn)改變了 (Foundation`_NSSetIntValueAndNotify)
dealloc 方法

最后理一下原理:

小結(jié):KVO 生成的子類 NSKVONotifying_Person 會(huì)重寫或添加了一些方法
1剥懒、是否是KVO監(jiān)聽類的方法 _isKVOA : 返回值是布爾類型 B(表示BOOL),值為YES
2合敦、重寫了需要監(jiān)聽的屬性的 setter 方法初橘,參數(shù)是 i(表示int) 類型,返回值是 v(表示void)
3充岛、重寫了 class 方法保檐,表示所屬的類是原來的 Person
4、重寫了 dealloc 方法

類方法

    Class kvoClass = objc_getClass("NSKVONotifying_Person");
    
    unsigned int classMethodCount = 0;
    // 使用元類
    Method *classMethods = class_copyMethodList(objc_getMetaClass(class_getName(kvoClass)), &classMethodCount);
    NSLog(@"classMethodCount: %d", classMethodCount);
    for (int m = 0; m < classMethodCount; m++) {
        Method method = classMethods[m];
        const char *name = sel_getName(method_getName(method));
        const char *typeEncoding = method_getTypeEncoding(method);
        NSLog(@"分割線 --------------");
        NSLog(@"methodName: %s", name);
        NSLog(@"methodTypeEncoding: %s", typeEncoding);
    }

輸出:

classMethodCount: 0

小結(jié):
KVO生成的子類 NSKVONotifying_Person 沒有類方法

斷點(diǎn)查看

image
  • KVO 生成的子類 NSKVONotifying_Person :
    • isa: NSKVONotifying_Person
    • class 方法: NSKVONotifying_Person
  • KVO 原來的類 Person
    • isa: NSKVONotifying_Person崔梗,這個(gè)是實(shí)際的類
    • class 方法: Person夜只,返回的是原來的類 Person,讓我們誤以為還是原來的類蒜魄,其實(shí)不是的扔亥,通過isa可以看到

屬性值變化的監(jiān)聽


    NSLog(@"---");
    obj.ageForNone = 10;
    
    NSLog(@"1111111");
    obj.ageForNone = 20;
    
    NSLog(@"2222222");
    obj.ageForNone = 20;
    
    // 監(jiān)聽方法
    - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    NSLog(@"keyPath: %@", keyPath);
    NSLog(@"object: %@", object);
    NSLog(@"change: %@", change);

輸出:

     keyPath: ageForNone
     object: <MyObject: 0x608000012ca0>
     change: {
     kind = 1;
     new = 10;
     old = 0;
     }
     1111111
     keyPath: ageForNone
     object: <MyObject: 0x608000012ca0>
     change: {
     kind = 1;
     new = 20;
     old = 10;
     }
     2222222
     keyPath: ageForNone
     object: <MyObject: 0x608000012ca0>
     change: {
     kind = 1;
     new = 20;
     old = 20;
     }

使用斷點(diǎn)進(jìn)入查看如下:

Foundation`_NSSetIntValueAndNotify:
    
    .......
    
    0x104511ff3 <+101>: movq   0x2ee306(%rip), %rsi      ; "willChangeValueForKey:"
    
    ......
    
    0x104512010 <+130>: callq  0x1046bcc78               ; symbol stub for: class_getMethodImplementation
    
    ......
    
    0x104512020 <+146>: movq   0x2ee2f1(%rip), %rsi      ; "didChangeValueForKey:"
    
    ......
    
    0x104512072 <+228>: movq   0x2eef17(%rip), %rsi      ; "_changeValueForKey:key:key:usingBlock:"

Foundation`NSKeyValueNotifyObserver:

    ......
    
    0x1044836ec <+41>:  movq   0x37d91d(%rip), %rdx      ; "_observeValueForKeyPath:ofObject:changeKind:oldValue:newValue:indexes:context:"
    0x1044836f3 <+48>:  movq   0x37c50e(%rip), %rsi      ; "respondsToSelector:"

    ......
    
    0x104483724 <+97>:  movq   0x37d8e5(%rip), %rsi      ; "_observeValueForKeyPath:ofObject:changeKind:oldValue:newValue:indexes:context:"

原來的類 Person 的一些特性

成員變量

    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([self.person class], &count);
    NSLog(@"count: %@", @(count));
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        const char *name = ivar_getName(ivar);
        ptrdiff_t offset = ivar_getOffset(ivar);
        const char *type = ivar_getTypeEncoding(ivar);
        NSLog(@"======");
        NSLog(@"name: %s", name);
        NSLog(@"offset: %td", offset);
        NSLog(@"type: %s", type);
    }

輸出結(jié)果如下:

count: 1
======
name: _age
offset: 8
type: i

小結(jié)
1、KVO 生成的子類 NSKVONotifying_Person 沒有額外的成員變量
2谈为、原來的類 Person 有一個(gè)成員變量 _age

屬性

    unsigned int propertyCount = 0;
    objc_property_t *properties = class_copyPropertyList([self.person class], &propertyCount);
    NSLog(@"propertyCount: %d", propertyCount);
    for (int j = 0; j < propertyCount ; j++) {
        objc_property_t property = properties[j];
        const char *name = property_getName(property);
        const char *attributes = property_getAttributes(property);
        NSLog(@"======");
        NSLog(@"propertyAttributes: %s", attributes);
        NSLog(@"propertyName: %s", name);
    }

輸出結(jié)果如下:

propertyCount: 1
======
propertyAttributes: Ti,N,V_age
propertyName: age

小結(jié):
1旅挤、KVO 生成的子類 NSKVONotifying_Person,沒有添加額外的屬性
2伞鲫、原來的類 Person 有一個(gè)屬性 age

實(shí)例方法

    unsigned int methodCount = 0;
    Method *methods = class_copyMethodList([self.person class], &methodCount);
    NSLog(@"methodCount: %d", methodCount);
    for (int k = 0; k < methodCount; k++) {
        Method method = methods[k];
        const char *name = sel_getName(method_getName(method));
        const char *typeEncoding = method_getTypeEncoding(method);
        NSLog(@"分割線 --------------");
        NSLog(@"methodName: %s", name);
        NSLog(@"methodTypeEncoding: %s", typeEncoding);
    }

輸出結(jié)果如下:

methodCount: 2
分割線 --------------
methodName: setAge:
methodTypeEncoding: v20@0:8i16
分割線 --------------
methodName: age
methodTypeEncoding: i16@0:8

小結(jié):
1粘茄、KVO 生成的子類 NSKVONotifying_Person 多了四個(gè)實(shí)例方法

  • setAge:: 參數(shù)為 int,返回值為 void
  • class: 沒有參數(shù)秕脓,返回值為 #驹闰,表示 class,也就是重寫了 class 方法
  • _isKVOA: 沒有參數(shù)撒会,返回值類型為 BOOL
  • dealloc: 重寫了 dealloc 方法
    2嘹朗、原來的類 Person 有兩個(gè)屬性的 settergetter 方法
  • setAge:
  • age
    unsigned int classMethodCount = 0;
    // 使用元類
    Method *classMethods = class_copyMethodList(objc_getMetaClass(class_getName([self.person class])), &classMethodCount);
    NSLog(@"classMethodCount: %d", classMethodCount);
    for (int m = 0; m < classMethodCount; m++) {
        Method method = classMethods[m];
        const char *name = sel_getName(method_getName(method));
        const char *typeEncoding = method_getTypeEncoding(method);
        NSLog(@"分割線 --------------");
        NSLog(@"methodName: %s", name);
        NSLog(@"methodTypeEncoding: %s", typeEncoding);
    }

輸出結(jié)果如下:

classMethodCount: 0

小結(jié):
1、KVO生成的子類 NSKVONotifying_Person 沒有類方法
2诵肛、原來的類 Person 沒有類方法

總結(jié)

原來的類Person KVO 生成的子類 NSKVONotifying_Person
class(對(duì)象方法) Person Person
class(類方法) Person NSKVONotifying_Person
isa(對(duì)象) NSKVONotifying_Person NSKVONotifying_Person
superclass NSObject Person
成員變量Ivar _age 沒有
屬性property age 沒有
實(shí)例方法method setAge: age setAge: class _isKVOA dealloc
類方法method 沒有 沒有
image
image
image

假設(shè)需要監(jiān)聽的類為MyObject:

  • KVO會(huì)自動(dòng)創(chuàng)建一個(gè)繼承于原來的類MyObject的子類NSKVONotifying_MyObject(runtime動(dòng)態(tài)創(chuàng)建一個(gè)類)
  • 把原來的類的對(duì)象isa指針指向新建的子類 NSKVONotifying_MyObject(runtime修飾isa指針的函數(shù))
  • 子類NSKVONotifying_MyObject會(huì)添加方法_isKVOA屹培,返回值YES表示是KVO監(jiān)聽的類
  • 子類NSKVONotifying_MyObject會(huì)添加方法class默穴,返回值MyObject表示是對(duì)象所屬于的類MyObject
  • 子類會(huì)重寫監(jiān)聽的屬性的setter方法(setAgeForNone:),方法內(nèi)部調(diào)用的是 Foundation框架中的_NSSetIntValueAndNotify函數(shù)
  • set值之前調(diào)用willChangeValueForKey:
  • set值之后調(diào)用didChangeValueForKey:
KVO前后

參考文章

[https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/KeyValueObserving/KeyValueObserving.html#//apple_ref/doc/uid/10000177-BCICJDHA]
(https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/KeyValueObserving/KeyValueObserving.html#//apple_ref/doc/uid/10000177-BCICJDHA)

https://tech.glowing.com/cn/implement-kvo/

fackbook第KVO的封裝

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末褪秀,一起剝皮案震驚了整個(gè)濱河市蓄诽,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌媒吗,老刑警劉巖仑氛,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異闸英,居然都是意外死亡锯岖,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門甫何,熙熙樓的掌柜王于貴愁眉苦臉地迎上來出吹,“玉大人,你說我怎么就攤上這事辙喂〈防危” “怎么了?”我有些...
    開封第一講書人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵巍耗,是天一觀的道長(zhǎng)秋麸。 經(jīng)常有香客問我,道長(zhǎng)炬太,這世上最難降的妖魔是什么灸蟆? 我笑而不...
    開封第一講書人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮娄琉,結(jié)果婚禮上次乓,老公的妹妹穿的比我還像新娘。我一直安慰自己孽水,他們只是感情好票腰,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著女气,像睡著了一般杏慰。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上炼鞠,一...
    開封第一講書人閱讀 49,111評(píng)論 1 285
  • 那天缘滥,我揣著相機(jī)與錄音,去河邊找鬼谒主。 笑死朝扼,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的霎肯。 我是一名探鬼主播擎颖,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼榛斯,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了搂捧?” 一聲冷哼從身側(cè)響起驮俗,我...
    開封第一講書人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎允跑,沒想到半個(gè)月后王凑,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡聋丝,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年索烹,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片潮针。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡术荤,死狀恐怖倚喂,靈堂內(nèi)的尸體忽然破棺而出每篷,到底是詐尸還是另有隱情,我是刑警寧澤端圈,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布焦读,位于F島的核電站,受9級(jí)特大地震影響舱权,放射性物質(zhì)發(fā)生泄漏矗晃。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一宴倍、第九天 我趴在偏房一處隱蔽的房頂上張望张症。 院中可真熱鬧,春花似錦鸵贬、人聲如沸俗他。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽兆衅。三九已至,卻和暖如春嗜浮,著一層夾襖步出監(jiān)牢的瞬間羡亩,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工危融, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留畏铆,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓吉殃,卻偏偏與公主長(zhǎng)得像辞居,于是被迫代替她去往敵國(guó)和親片排。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

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

  • KVO(Key-value observing)提供一種在其它對(duì)象的屬性更改時(shí)通知觀察它的對(duì)象的一種機(jī)制速侈。當(dāng)然它和...
    barry閱讀 408評(píng)論 0 1
  • 導(dǎo)語: KVO全稱Key Value Observing率寡,直譯為鍵值觀察。KVO 作為 iOS 中一種強(qiáng)大并且有效...
    xianminxiao閱讀 1,108評(píng)論 0 2
  • 提到KVC/KVO大家一定不會(huì)陌生倚搬,用起來也很簡(jiǎn)單冶共,下面一起來探究一下它們的實(shí)現(xiàn)原理。 一每界、KVC 1捅僵、查找過程 ...
    Tamp_閱讀 1,024評(píng)論 7 12
  • 概念 基本使用 觸發(fā)模式 屬性依賴 容器類的使用 自定義KVO 概念 KVO全稱Key-Value Observi...
    it_Xiong閱讀 677評(píng)論 0 2
  • KVC[http://www.reibang.com/p/2293b2ee5b9a] KVC的全程是 key-v...
    福爾摩羅閱讀 182評(píng)論 0 3