kvo

http://blog.sina.com.cn/s/blog_be8740510102xgno.html 概念
http://www.cocoachina.com/ios/20150313/11321.html 簡單自定義
http://blog.csdn.net/kesalin/article/details/8194240# 深入理解
http://www.itdecent.cn/p/35029ec508a3
http://www.itdecent.cn/p/f900de4a1495 runtime 基礎知識
http://www.cocoachina.com/ios/20141106/10150.html
剛開始時 我覺得kvo 我需要大量的書寫一部分內(nèi)容,但是在我查閱了部分資料以后,我其實感覺
官方的解釋其實很到位


當你觀察一個對象時,一個新的類會動態(tài)被創(chuàng)建。這個類繼承自該對象的原本的類,并重寫了被觀察屬性的 setter 方法。自然,重寫的 setter 方法會負責在調(diào)用原 setter 方法之前和之后,通知所有觀察對象值的更改。最后把這個對象的 isa 指針 ( isa 指針告訴 Runtime 系統(tǒng)這個對象的類是什么 ) 指向這個新創(chuàng)建的子類,對象就神奇的變成了新創(chuàng)建的子類的實例。

原來,這個中間類,繼承自原本的那個類。不僅如此,Apple 還重寫了 -class 方法,企圖欺騙我們這個類沒有變,就是原本那個類。更具體的信息,去跑一下 Mike Ash 的那篇文章里的代碼就能明白,這里就不再重復。


看起來感覺很簡單碼 ,就是產(chǎn)生一個派生類,然后重寫并且指向這個派生累的set方法,感覺一句話呢 ,但是感覺 這句話是在是不接地氣.不接地氣的原因就是我們隊runtime的了解太少
那么好 ,我們先說一下呢 簡單的 產(chǎn)生一個派生類

Class clazz = object_getClass(self);

然后給這個派生類覆蓋一下set方法,那么這里需要解釋一下
kvo 設置監(jiān)聽的方法是在nsobject 那么覆蓋set方法就是

class_addMethod(clazz, setterSelector, (IMP)kvo_setter, types);

這一句的方法就是 給這個clazz類 添加一個名字是setterSelector的方法 實現(xiàn)這個方法的指針指向 kvo_setter方法的指針.
那么也就是說 kvo的工做分兩部分做 一部分是在監(jiān)聽是做的 一部分是在 重寫的set方法里面的做的
這里是在網(wǎng)上找到的大神仿寫的kvo

- (void)PG_addObserver:(NSObject *)observer
                forKey:(NSString *)key
             withBlock:(PGObservingBlock)block
{
    SEL setterSelector = NSSelectorFromString(setterForGetter(key));//更具key(也就是我們平時的path),用它生成一個方法(通過指真可以找到他)

    Method setterMethod = class_getInstanceMethod([self class], setterSelector);
//這里的 self 指的就是被監(jiān)聽類,這一句就是得到監(jiān)聽類里面是否有相應屬性
    if (!setterMethod) {
        NSString *reason = [NSString stringWithFormat:@"Object %@ does not have a setter for key %@", self, key];
        @throw [NSException exceptionWithName:NSInvalidArgumentException
                                       reason:reason
                                     userInfo:nil];
        //如果沒有這拋出異常
        return;
    }
    
    Class clazz = object_getClass(self);
//我理解的就是到這里產(chǎn)生一個新的類也就是派生類
    NSString *clazzName = NSStringFromClass(clazz);
    //得到這個類的名字
    // if not an KVO class yet
    if (![clazzName hasPrefix:kPGKVOClassPrefix]) {
        clazz = [self makeKvoClassWithOriginalClassName:clazzName];
        object_setClass(self, clazz);
//這里應該就是修改了派生累的名字
    }
    
    // add our kvo setter if this class (not superclasses) doesn't implement the setter?
    if (![self hasSelector:setterSelector]) {
        const char *types = method_getTypeEncoding(setterMethod);
        class_addMethod(clazz, setterSelector, (IMP)kvo_setter, types);
//給這個派生類添加一個set 方法  并且指向kvo_setter這個方法 也就是說沒次修改屬性時 在調(diào)用set方法時 都會調(diào)用這個方法
    }
    
    PGObservationInfo *info = [[PGObservationInfo alloc] initWithObserver:observer Key:key block:block];
//這里大神對kvo進行了優(yōu)化 采用了 block  其實就是一個model   將這三個參數(shù)復制而已
    NSMutableArray *observers = objc_getAssociatedObject(self, (__bridge const void *)(kPGKVOAssociatedObservers));
//這里呢就是在self中查找一個關聯(lián)的數(shù)組
    if (!observers) {
        observers = [NSMutableArray array];
        objc_setAssociatedObject(self, (__bridge const void *)(kPGKVOAssociatedObservers), observers, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
//如果之前沒關聯(lián)那么關聯(lián)上,(意思就是數(shù)組成為了self的一個屬性把)
    [observers addObject:info];
//把model添加到數(shù)組里面
}


這里就是前面提到派生類被重寫的set方法

static void kvo_setter(id self, SEL _cmd, id newValue)
//其實說的就是_cmd在Objective-C的方法中表示當前方法的selector,正如同self表示當前方法調(diào)用的對象實例一樣。
{
    NSString *setterName = NSStringFromSelector(_cmd);
    NSString *getterName = getterForSetter(setterName);
    
    if (!getterName) {
        NSString *reason = [NSString stringWithFormat:@"Object %@ does not have setter %@", self, setterName];
        @throw [NSException exceptionWithName:NSInvalidArgumentException
                                       reason:reason
                                     userInfo:nil];
        return;
    }
    
    id oldValue = [self valueForKey:getterName];
    
    struct objc_super superclazz = {
        .receiver = self,
        .super_class = class_getSuperclass(object_getClass(self))
    };
    
    // cast our pointer so the compiler won't complain
    void (*objc_msgSendSuperCasted)(void *, SEL, id) = (void *)objc_msgSendSuper;
    
    // call super's setter, which is original class's setter method
    objc_msgSendSuperCasted(&superclazz, _cmd, newValue);
    
    // look up observers and call the blocks
    NSMutableArray *observers = objc_getAssociatedObject(self, (__bridge const void *)(kPGKVOAssociatedObservers));
    for (PGObservationInfo *each in observers) {
        if ([each.key isEqualToString:getterName]) {
            dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
                each.block(self, getterName, oldValue, newValue);
            });
        }
    }
}



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

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

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