iOS中block的循環引用問題

本文主要介紹ARC下block的循環引用問題,舉例說明引起循環引用的場景和相應的解決方案。

在講block的循環引用問題之前,我們需要先了解一下iOS的內存管理機制和block的基本知識

iOS的內存管理機制

Objective-C在iOS中不支持GC(垃圾回收)機制,而是采用的引用計數的方式管理內存。

引用計數(Reference Count)

在引用計數中,每一個對象負責維護對象所有引用的計數值。當一個新的引用指向對象時,引用計數器就遞增,當去掉一個引用時,引用計數就遞減。當引用計數到零時,該對象就將釋放占有的資源。

我們通過開關房間的燈為例來?說明引用計數機制。


引用《Pro Multithreading and Memory Management for iOS and OS X》中的圖片

圖中,“需要照明的人數”即對應我們要說的引用計數值。

  1. 第一個人進入辦公室,“需要照明的人數”加1,計數值從0變為1,因此需要開燈;
  2. 之后每當有人進入辦公室,“需要照明的人數”就加1。如計數值從1變成2;
  3. 每當有人下班離開辦公室,“需要照明的人數”加減1如計數值從2變成1;
  4. 最后一個人下班離開辦公室時,“需要照明的人數”減1。計數值從1變成0,因此需要關燈。

在Objective-C中,”對象“相當于辦公室的照明設備,”對象的使用環境“相當于進入辦公室的人。上班進入辦公室的人對辦公室照明設備發出的動作,與Objective-C中的對應關系如下表

對照明設備所做的動作 對Objective-C對象所做的動作
開燈 生成對象
需要照明 持有對象
不需要照明 釋放對象
關燈 廢棄對象

使用計數功能計算需要照明的人數,使辦公室的照明得到了很好的管理。同樣,使用引用計數功能,對象也就能得到很好的管理,這就是Objective-C內存管理,如下圖所示

引用《Pro Multithreading and Memory Management for iOS and OS X》中的圖片

MRC(Manual Reference Counting)中引起應用計數變化的方法

Objective-C對象方法 說明
alloc/new/copy/mutableCopy 創建對象,引用計數加1
retain 引用計數加1
release 引用計數減1
dealloc 當引用計數為0時調用
[NSArray array] 引用計數不增加,由自動釋放池管理
[NSDictionary dictionary] 引用計數不增加,由自動釋放池管理

自動釋放池

關于自動釋放,不是本文的重點,這里就不講了。

ARC(Automatic Reference Counting)中內存管理

Objective-C對象所有權修飾符 說明
__strong 對象默認修飾符,對象強引用,在對象超出作用域時失效。其實就相當于retain操作,超出作用域時執行release操作
__weak 弱引用,不持有對象,對象釋放時會將對象置nil。
__unsafe_unretained 弱引用,不持有對象,對象釋放時不會將對象置nil。
__autoreleasing 自動釋放,由自動釋放池管理對象

block的基本知識

block的基本知識這里就不細說了,可以看看我的文章說說Objective-C中的block

循環引用問題

兩個對象相互持有,這樣就會造成循環引用,如下圖所示

兩個對象相互持有

圖中,對象A持有對象B,對象B持有對象A,相互持有,最終導致兩個對象都不能釋放。

block中循環引用問題

由于block會對block中的對象進行持有操作,就相當于持有了其中的對象,而如果此時block中的對象又持有了該block,則會造成循環引用。如下,

typedef void(^block)();

@property (copy, nonatomic) block myBlock;
@property (copy, nonatomic) NSString *blockString;

- (void)testBlock {
    self.myBlock = ^() {
        //其實注釋中的代碼,同樣會造成循環引用
        NSString *localString = self.blockString;
        //NSString *localString = _blockString;
        //[self doSomething];
    };
}

注:以下調用注釋掉的代碼同樣會造成循環引用,因為不管是通過self.blockString還是_blockString,或是函數調用[self doSomething],因為只要 block中用到了對象的屬性或者函數,block就會持有該對象而不是該對象中的某個屬性或者函數。

當有someObj持有self對象,此時的關系圖如下。


當someObj對象release self對象時,self和myblock相互引用,retainCount都為1,造成循環引用

解決方法:

__weak typeof(self) weakSelf = self;
self.myBlock = ^() {
    NSString *localString = weakSelf.blockString;
};

使用__weak修飾self,使其在block中不被持有,打破循環引用。開始狀態如下

當someObj對象釋放self對象時,Self的retainCount為0,走dealloc,釋放myBlock對象,使其retainCount也為0。

其實以上循環引用的情況很容易發現,因為此時Xcode就會報警告。而發生在多個對象間的時候,Xcode就檢測不出來了,這往往就容易被忽略。

//ClassB
@interface ClassB : NSObject
@property (strong, nonatomic) ClassA *objA;
- (void)doSomething;
@end
  
//ClassA
@property (strong, nonatomic) ClassB *objB;
@property (copy, nonatomic) block myBlock;

- (void)testBlockRetainCycle {
    ClassB* objB = [[ClassB alloc] init];
    self.myBlock = ^() {
        [objB doSomething];
    };
    objB.objA = self;
}

解決方法:

- (void)testBlockRetainCycle {
    ClassB* objB = [[ClassB alloc] init];
    __weak typeof(objB) weakObjB = objB;
    self.myBlock = ^() {
        [weakObjB doSomething];
    };
    objB.objA = self;
}

將objA對象weak,使其不在block中被持有

注:以上使用__weak打破循環的方法只在ARC下才有效,在MRC下應該使用__block

或者,在block執行完后,將block置nil,這樣也可以打破循環引用

- (void)testBlockRetainCycle {
    ClassB* objB = [[ClassB alloc] init];
    self.myBlock = ^() {
        [objB doSomething];
    };
    objA.objA = self;
    self.myBlock();
    self.myBlock = nil;
}

這樣做的缺點是,block只會執行一次,因為block被置nil了,要再次使用的話,需要重新賦值。

一些不會造成循環引用的block

在開發工程中,發現一些同學并沒有完全理解循環引用,以為只要有block的地方就會要用__weak來修飾對象,這樣完全沒有必要,以下幾種block是不會造成循環引用的。

大部分GCD方法

dispatch_async(dispatch_get_main_queue(), ^{
    [self doSomething];
});

因為self并沒有對GCD的block進行持有,沒有形成循環引用。目前我還沒碰到使用GCD導致循環引用的場景,如果某種場景self對GCD的block進行了持有,則才有可能造成循環引用。

block并不是屬性值,而是臨時變量

- (void)doSomething {
    [self testWithBlock:^{
        [self test];
    }];
}

- (void)testWithBlock:(void(^)())block {
    block();
}

- (void)test {
    NSLog(@"test");
}

這里因為block只是一個臨時變量,self并沒有對其持有,所以沒有造成循環引用

block使用對象被提前釋放

看下面例子,有這種情況,如果不只是ClassA持有了myBlock,ClassB也持有了myBlock。

當ClassA被someObj對象釋放后

此時,ClassA對象已經被釋放,而myBlock還是被ClassB持有,沒有釋放;如果myBlock這個時被調度,而此時ClassA已經被釋放,此時訪問的ClassA將是一個nil對象(使用__weak修飾,對象釋放時會置為nil),而引發錯誤。

另一個常見錯誤使用是,開發者擔心循環引用錯誤(如上所述不會出現循環引用的情況),使用__weak。比如

__weak typeof(self) weakSelf = self;
dispatch_async(dispatch_get_main_queue(), ^{
    [weakSelf doSomething];
});

因為將block作為參數傳給dispatch_async時,系統會將block拷貝到堆上,而且block會持有block中用到的對象,因為dispatch_async并不知道block中對象會在什么時候被釋放,為了確保系統調度執行block中的任務時其對象沒有被意外釋放掉,dispatch_async必須自己retain一次對象(即self),任務完成后再release對象(即self)。但這里使用__weak,使dispatch_async沒有增加self的引用計數,這使得在系統在調度執行block之前,self可能已被銷毀,但系統并不知道這個情況,導致block執行時訪問已經被釋放的self,而達不到預期的結果。

注:如果是在MRC模式下,使用__block修飾self,則此時block訪問被釋放的self,則會導致crash。

該場景下的代碼

// ClassA.m
- (void)test {
    __weak MyClass* weakSelf = self;
    double delayInSeconds = 10.0f;
    dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
    dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
        NSLog(@"%@", weakSelf);
    });
}

// ClassB.m
- (void)doSomething {
    NSLog(@"do something");
    ClassA *objA = [[ClassA alloc] init];
    [objA test];
}

運行結果

[5988:435396] do something
[5988:435396] self:(null)

解決方法:

對于這種場景,就不應該使用__weak來修飾對象,讓dispatch_after對self進行持有,保證block執行時self還未被釋放。

block執行過程中對象被釋放

還有一種場景,在block執行開始時self對象還未被釋放,而執行過程中,self被釋放了,此時訪問self時,就會發生錯誤。

對于這種場景,應該在block中對 對象使用__strong修飾,使得在block期間對 對象持有,block執行結束后,解除其持有。

- (void)testBlockRetainCycle {
    ClassA* objA = [[ClassA alloc] init];
    __weak typeof(objA) weakObjA = objA;
    self.myBlock = ^() {
        __strong typeof(weakObjA) strongWeakObjA = weakObjA;
        [strongWeakObjA doSomething];
    };
    objA.objA = self;
}

注:此方法只能保證在block執行期間對象不被釋放,如果對象在block執行執行之前已經被釋放了,該方法也無效。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,406評論 6 538
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,034評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,413評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,449評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,165評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,559評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,606評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,781評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,327評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,084評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,278評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,849評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,495評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,927評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,172評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,010評論 3 396
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,241評論 2 375

推薦閱讀更多精彩內容