『ios』kvo中安全的移除監(jiān)聽(tīng)

是否經(jīng)常在項(xiàng)目中遇到kvo移除崩潰的錯(cuò)誤?
其實(shí)我們可以用try catch來(lái)解決這個(gè)問(wèn)題。
今天看到了二種比較優(yōu)雅的解決辦法。

+ (void)load
{
    [self switchMethod];
}

+ (void)switchMethod
{
    SEL removeSel = @selector(removeObserver:forKeyPath:);
    SEL myRemoveSel = @selector(removeSafe:forKeyPath:);
    SEL addSel = @selector(addObserver:forKeyPath:options:context:);
    SEL myaddSel = @selector(addSafe:forKeyPath:options:context:);
    
    Method systemRemoveMethod = class_getClassMethod([self class],removeSel);
    Method DasenRemoveMethod = class_getClassMethod([self class], myRemoveSel);
    Method systemAddMethod = class_getClassMethod([self class],addSel);
    Method DasenAddMethod = class_getClassMethod([self class], myaddSel);
    
    method_exchangeImplementations(systemRemoveMethod, DasenRemoveMethod);
    method_exchangeImplementations(systemAddMethod, DasenAddMethod);
}

// 交換后的方法
- (void)removeSafe:(NSObject *)observer forKeyPath:(NSString *)keyPath
{
    if ([self observerKeyPath:keyPath observer:observer]) {
        [self removeSafe:observer forKeyPath:keyPath];
    }
}

// 交換后的方法
- (void)addSafe:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
{
    if (![self observerKeyPath:keyPath observer:observer]) {
        [self addSafe:observer forKeyPath:keyPath options:options context:context];
    }
}


// 進(jìn)行檢索獲取Key
- (BOOL)observerKeyPath:(NSString *)key observer:(id )observer
{
    id info = self.observationInfo;
    NSArray *array = [info valueForKey:@"_observances"];
    for (id objc in array) {
        id Properties = [objc valueForKeyPath:@"_property"];
        id newObserver = [objc valueForKeyPath:@"_observer"];
        
        NSString *keyPath = [Properties valueForKeyPath:@"_keyPath"];
        if ([key isEqualToString:keyPath] && [newObserver isEqual:observer]) {
            return YES;
        }
    }
    return NO;
}

我們可以利用

    id info = self.observationInfo;
    NSArray *array = [info valueForKey:@"_observances"];

拿到當(dāng)前類進(jìn)行kvo監(jiān)聽(tīng)到對(duì)象拍柒。如下圖所示

image.png

然后通過(guò)對(duì)比監(jiān)聽(tīng)到key和監(jiān)聽(tīng)的對(duì)象是否相同。

 id Properties = [objc valueForKeyPath:@"_property"];
        id newObserver = [objc valueForKeyPath:@"_observer"];
image.png

還有一種解決方案屈暗。

@interface ObserverData : NSObject
@property (nonatomic, strong)id objc;
@property (nonatomic, copy)  NSString *keyPath;
- (instancetype)initWithObjc:(id)objc key:(NSString *)key;

@end
@implementation ObserverData
- (instancetype)initWithObjc:(id)objc key:(NSString *)key
{
    if (self = [super init]) {
        self.objc = objc;
        self.keyPath = key;
    }
    return self;
}
@end

#import "DSObserver.h"
@implementation DSObserver
+ (instancetype)sharedDSObserver
{
    static id objc;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        objc = [NSMutableArray array];
    });
    return objc;
}
@end


#pragma mark - 第二種方案拆讯,利用私有屬性
// 交換后的方法
- (void)removeSafe:(NSObject *)observer forKeyPath:(NSString *)keyPath
{
    NSMutableArray *Observers = [DSObserver sharedDSObserver];
    ObserverData *userPathData = [self observerKeyPath:keyPath];
    // 如果有該key值那么進(jìn)行刪除
    if (userPathData) {
        [Observers removeObject:userPathData];
        [self removeSafe:observer forKeyPath:keyPath];
    }
    return;
}

// 交換后的方法
- (void)addSafe:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
{
    ObserverData *userPathData= [[ObserverData alloc]initWithObjc:self key:keyPath];
    NSMutableArray *Observers = [DSObserver sharedDSObserver];

    // 如果沒(méi)有注冊(cè),那么才進(jìn)行注冊(cè)
    if (![self observerKeyPath:keyPath]) {
        [Observers addObject:userPathData];
        [self addSafe:observer forKeyPath:keyPath options:options context:context];
    }

}
//
// 進(jìn)行檢索养叛,判斷是否已經(jīng)存儲(chǔ)了該Key值
- (ObserverData *)observerKeyPath:(NSString *)keyPath
{
    NSMutableArray *Observers = [DSObserver sharedDSObserver];
    for (ObserverData *data in Observers) {
        if ([data.objc isEqual:self] && [data.keyPath isEqualToString:keyPath]) {
            return data;
        }
    }
    return nil;
}

可以新建一個(gè)全局的管理kvo監(jiān)聽(tīng)管理者數(shù)組种呐,然后如果監(jiān)聽(tīng)就add,移除就remove弃甥,進(jìn)行之前爽室,進(jìn)行判斷是否已經(jīng)存在或者是已經(jīng)移除,來(lái)避免問(wèn)題淆攻。
其實(shí)我比較推薦第二種解決方案阔墩。因?yàn)榈谝环N應(yīng)該會(huì)有一些特殊情況不能完全覆蓋到。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末瓶珊,一起剝皮案震驚了整個(gè)濱河市啸箫,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌伞芹,老刑警劉巖忘苛,帶你破解...
    沈念sama閱讀 216,402評(píng)論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異丑瞧,居然都是意外死亡柑土,警方通過(guò)查閱死者的電腦和手機(jī)蜀肘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)绊汹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人扮宠,你說(shuō)我怎么就攤上這事西乖『疲” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,483評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵获雕,是天一觀的道長(zhǎng)薄腻。 經(jīng)常有香客問(wèn)我,道長(zhǎng)届案,這世上最難降的妖魔是什么庵楷? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,165評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮楣颠,結(jié)果婚禮上尽纽,老公的妹妹穿的比我還像新娘。我一直安慰自己童漩,他們只是感情好弄贿,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著矫膨,像睡著了一般差凹。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上侧馅,一...
    開(kāi)封第一講書(shū)人閱讀 51,146評(píng)論 1 297
  • 那天危尿,我揣著相機(jī)與錄音,去河邊找鬼馁痴。 笑死脚线,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的弥搞。 我是一名探鬼主播邮绿,決...
    沈念sama閱讀 40,032評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼攀例!你這毒婦竟也來(lái)了船逮?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,896評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤粤铭,失蹤者是張志新(化名)和其女友劉穎挖胃,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體梆惯,經(jīng)...
    沈念sama閱讀 45,311評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡酱鸭,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評(píng)論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了垛吗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片凹髓。...
    茶點(diǎn)故事閱讀 39,696評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖怯屉,靈堂內(nèi)的尸體忽然破棺而出蔚舀,到底是詐尸還是另有隱情饵沧,我是刑警寧澤,帶...
    沈念sama閱讀 35,413評(píng)論 5 343
  • 正文 年R本政府宣布赌躺,位于F島的核電站狼牺,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏礼患。R本人自食惡果不足惜是钥,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評(píng)論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望缅叠。 院中可真熱鬧咏瑟,春花似錦、人聲如沸痪署。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)狼犯。三九已至余寥,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間悯森,已是汗流浹背宋舷。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,815評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瓢姻,地道東北人祝蝠。 一個(gè)月前我還...
    沈念sama閱讀 47,698評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像幻碱,于是被迫代替她去往敵國(guó)和親绎狭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評(píng)論 2 353