iOS中objc_setAssociatedObject關(guān)聯(lián)對象自動置空

前言

有經(jīng)驗(yàn)的iOS開發(fā)者都知道玖姑,ARC中的weak關(guān)鍵字可以在對象銷毀時(shí) 指針自動置成nil缎谷,在OC中向nil發(fā)消息是安全的,所以不會造成野指針錯(cuò)誤蚁孔。

在category中擴(kuò)展屬性時(shí)屑埋,一般會使用runtime的關(guān)聯(lián)對象(AssociatedObject)技術(shù)豪筝,關(guān)聯(lián)對象的策略(Policy)有5個(gè):

OBJC_ASSOCIATION_ASSIGN = 0, //弱引用
OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,//強(qiáng)引用,非原子性
OBJC_ASSOCIATION_COPY_NONATOMIC = 3,//copy摘能,非原子性
OBJC_ASSOCIATION_RETAIN = 01401,//強(qiáng)引用续崖,原子性
OBJC_ASSOCIATION_COPY = 01403//copy,原子性

我們可以發(fā)現(xiàn)团搞,在5個(gè)策略中并沒有weak類型严望,OBJC_ASSOCIATION_ASSIGN 策略雖然可以弱引用,但是在對象銷毀的時(shí)候不能自動將指針置nil逻恐。

現(xiàn)象舉例

我們使用Category給UIViewController擴(kuò)展一個(gè)UILabel類型的屬性aLabel像吻,并使用OBJC_ASSOCIATION_ASSIGN 策略,看看會發(fā)生什么复隆。代碼如下:

//  UIViewController+Category.h
@interface UIViewController (Category)

@property (nonatomic, strong) UILabel *aLabel;

@end

//  UIViewController+Category.m
@implementation UIViewController (Category)

- (void)setALabel:(UILabel *)aLabel {
    objc_setAssociatedObject(self, @selector(aLabel), aLabel, OBJC_ASSOCIATION_ASSIGN);
}

- (UILabel *)aLabel {
    return objc_getAssociatedObject(self, @selector(aLabel));
}

@end

然后賦值使用:

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor orangeColor];
    
    UILabel *label = [[UILabel alloc] init];
    self.aLabel = label;
    
    NSLog(@"-viewDidLoad-\nself.aLabel = %@", self.aLabel);
}

- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    
    NSLog(@"-viewDidAppear-\nself.aLabel = %@", self.aLabel);
}


@end

在viewDidLoad方法中對aLabel進(jìn)行了賦值拨匆,出了viewDidLoad作用域后,aLabel指向的對象會被銷毀挽拂。運(yùn)行后惭每,我們發(fā)現(xiàn)程序崩潰了。打開僵尸對象調(diào)試亏栈,報(bào)錯(cuò)如下:

*** -[UILabel retain]: message sent to deallocated instance

對象被銷毀了台腥,指針沒有置nil宏赘,造成了崩潰。

解決方案

我們需要做的是在獲取到對象銷毀的時(shí)機(jī)览爵,然后將相應(yīng)的指針指向nil。如果是我們自己創(chuàng)建的類镇饮,可以在dealloc方法中進(jìn)行block回調(diào)蜓竹。但是系統(tǒng)早已創(chuàng)建好的類,開發(fā)者沒有地方可以寫dealloc回調(diào)储藐。

與蘋果系統(tǒng)對KVO的實(shí)現(xiàn)原理參考我這篇文章類似俱济,我們可以在屬性的set方法中,動態(tài)創(chuàng)建一個(gè)關(guān)聯(lián)對象的子類钙勃,重寫新類的dealloc方法蛛碌,在新類的dealloc中將指針置nil,并將關(guān)聯(lián)對象的isa指針指向新類辖源。

沿著這個(gè)思路蔚携,我們可以寫出以下代碼:

void objc_setAssociatedObject_weak(id _Nonnull object, const void * _Nonnull key, id _Nullable value) {
    
    //子類的名字
    NSString *name = [NSString stringWithFormat:@"AssociationWeak_%@", NSStringFromClass([value class])];
    Class class = objc_getClass(name.UTF8String);
    
    //如果子類不存在,動態(tài)創(chuàng)建子類
    if (!class) {
        class = objc_allocateClassPair([value class], name.UTF8String, 0);
        objc_registerClassPair(class);
    }
    
    SEL deallocSEL = NSSelectorFromString(@"dealloc");
    Method deallocMethod = class_getInstanceMethod([value class], deallocSEL);
    const char *types = method_getTypeEncoding(deallocMethod);
    
    //在子類dealloc方法中將object的指針置為nil
    IMP imp = imp_implementationWithBlock(^(id _s, int k) {
        objc_setAssociatedObject(object, key, nil, OBJC_ASSOCIATION_ASSIGN);
    });
    
    //添加子類的dealloc方法
    class_addMethod(class, deallocSEL, imp, types);
    
    //將value的isa指向動態(tài)創(chuàng)建的子類
    object_setClass(value, class);
    
    objc_setAssociatedObject(object, key, value, OBJC_ASSOCIATION_ASSIGN);
}

在進(jìn)行關(guān)聯(lián)對象的操作時(shí)克饶,我們使用自己新寫的方法酝蜒,不再使用系統(tǒng)關(guān)聯(lián)對象方法:

- (void)setALabel:(UILabel *)aLabel {
    objc_setAssociatedObject_weak(self, @selector(aLabel), aLabel);
    

再運(yùn)行程序看看打印結(jié)果:

-viewDidLoad-
self.aLabel = <AssociationWeak_UILabel: 0x7fd174f0b770; baseClass = UILabel; frame = (0 0; 0 0); userInteractionEnabled = NO; layer = <_UILabelLayer: 0x600000818780>>

-viewDidAppear-
self.aLabel = (null)

我們發(fā)現(xiàn)這樣成功捕捉到了對象被銷毀的時(shí)機(jī),并將指針指向了nil矾湃,沒有出現(xiàn)崩潰的情況亡脑。

至此,我們成功做到了弱引用對象銷毀后邀跃,指針自動置空的操作霉咨。我將方法封裝到了NSObject的分類中。任何繼承自NSObject的OC對象拍屑,在關(guān)聯(lián)對象時(shí)途戒,都可以使用。

DEMO下載

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末僵驰,一起剝皮案震驚了整個(gè)濱河市棺滞,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌矢渊,老刑警劉巖继准,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異矮男,居然都是意外死亡移必,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門毡鉴,熙熙樓的掌柜王于貴愁眉苦臉地迎上來崔泵,“玉大人秒赤,你說我怎么就攤上這事≡魅常” “怎么了入篮?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長幌甘。 經(jīng)常有香客問我潮售,道長,這世上最難降的妖魔是什么锅风? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任酥诽,我火速辦了婚禮,結(jié)果婚禮上皱埠,老公的妹妹穿的比我還像新娘肮帐。我一直安慰自己,他們只是感情好边器,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布训枢。 她就那樣靜靜地躺著,像睡著了一般忘巧。 火紅的嫁衣襯著肌膚如雪肮砾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天袋坑,我揣著相機(jī)與錄音仗处,去河邊找鬼。 笑死枣宫,一個(gè)胖子當(dāng)著我的面吹牛婆誓,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播也颤,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼洋幻,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了翅娶?” 一聲冷哼從身側(cè)響起文留,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎竭沫,沒想到半個(gè)月后燥翅,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蜕提,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年森书,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡凛膏,死狀恐怖杨名,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情猖毫,我是刑警寧澤台谍,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站吁断,受9級特大地震影響趁蕊,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜胯府,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一介衔、第九天 我趴在偏房一處隱蔽的房頂上張望恨胚。 院中可真熱鬧骂因,春花似錦、人聲如沸赃泡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽升熊。三九已至俄烁,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間级野,已是汗流浹背页屠。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蓖柔,地道東北人辰企。 一個(gè)月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像况鸣,于是被迫代替她去往敵國和親牢贸。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評論 2 354