iOS-38-ARC內存泄漏

block系列

在 ARC 下,當 block 獲取到外部變量時,由于編譯器無法預測獲取到的變量何時會被突然釋放,為了保證程序能夠正確運行,讓 block 持有獲取到的變量,向系統顯明:我要用它,你們千萬別把它回收了!然而,也正因 block 持有了變量,容易導致變量和 block 的循環引用,造成內存泄露!

對于 block 中的循環引用通常有兩種解決方法:

1、將對象置為 nil ,消除引用,打破循環引用;

(這種做法有個很明顯的缺點,即開發者必須保證 _networkFetecher = nil; 運行過。若不如此,就無法打破循環引用。

但這種做法的使用場景也很明顯,由于 block 的內存必須等待持有它的對象被置為 nil 后才會釋放。所以如果開發者希望自己控制 block 對象的生命周期時,就可以使用這種方法。)

2、將強引用轉換成弱引用,打破循環引用;

(__weak __typeof(self) weakSelf = self;如果想防止 weakSelf 被釋放,可以再次強引用 __typeof(&weakSelf) strongSelf = weakSelf;代碼 __typeof(&weakSelf) strongSelf 括號內為什么要加 &* 呢?主要是為了兼容早期的 LLVM

block 的內存泄露問題包括自定義的 block,系統框架的 block 如 GCD 等,都需要注意循環引用的問題。

有個值得一提的細節是,在種類眾多的 block 當中,方法名帶有 usingBlock 的 Cocoa Framework 方法或 GCD 的 API ,如

- enumerateObjectsUsingBlock:
- sortUsingComparator:

這一類 API 同樣會有循環引用的隱患,但原因并非編譯器做了保留,而是 API 本身會對傳入的 block 做一個復制的操作。

delegate系列

@property (nonatomic, weak) id  delegate;

說白了就是循環使用的問題,假如我們是寫的strong,那么 兩個類之間調用代理就是這樣的啦

BViewController *bViewController = [[BViewController alloc] init];
bViewController.delegate = self; //假設 self 是AViewController
[self.navigationController pushViewController:bViewController animated:YES];
/**
 假如是 strong 的情況
    bViewController.delegate ===> AViewController (也就是 A 的引用計數 + 1)
    AViewController 本身又是引用了  ===> delegate 引用計數 + 1
 導致: AViewController  Delegate ,也就循環引用啦
 */

Delegate創建并強引用了 AViewController;(strong ==> A 強引用、weak ==> 引用計數不變)
所以用 strong的情況下,相當于 Delegate 和 A 兩個互相引用啦,A 永遠會有一個引用計數 1 不會被釋放,所以造成了永遠不能被內存釋放,因此weak是必須的。

performSelector 系列

performSelector 顧名思義即在運行時執行一個 selector,最簡單的方法如下

- (id)performSelector:(SEL)selector;

這種調用 selector 的方法和直接調用 selector 基本等效,執行效果相同

[object methodName];
[object performSelector:@selector(methodName)];

但 performSelector 相比直接調用更加靈活

SEL selector;
if (/* some condition */) {
    selector = @selector(newObject);
} else if (/* some other condition */) {
    selector = @selector(copy);
} else {
    selector = @selector(someProperty);
}
id ret = [object performSelector:selector];

這段代碼就相當于在動態之上再動態綁定。在 ARC 下編譯這段代碼,編譯器會發出警告

warning: performSelector may cause a leak because its selector is unknow [-Warc-performSelector-leak]
正是由于動態,編譯器不知道即將調用的 selector 是什么,不了解方法簽名和返回值,甚至是否有返回值都不懂,所以編譯器無法用 ARC 的內存管理規則來判斷返回值是否應該釋放。因此,ARC 采用了比較謹慎的做法,不添加釋放操作,即在方法返回對象時就可能將其持有,從而可能導致內存泄露。

以本段代碼為例,前兩種情況(newObject, copy)都需要再次釋放,而第三種情況不需要。這種泄露隱藏得如此之深,以至于使用 static analyzer 都很難檢測到。如果把代碼的最后一行改成
    [object performSelector:selector];
不創建一個返回值變量測試分析,簡直難以想象這里居然會出現內存問題。所以如果你使用的 selector 有返回值,一定要處理掉。

還有一種情況就是performSelector的延時調用[self performSelector:@selector(method1:) withObject:self.myView afterDelay:5];,performSelector關于內存管理的執行原理是這樣的,當執行[self performSelector:@selector(method1:) withObject:self.myView afterDelay:5];的時候,系統將myView的引用計數加1,執行完這個方法之后將myView的引用計數減1,而在延遲調用的過程中很可能就會出現,這個方法被調用了,但是沒有執行,此時myView的引用計數并沒有減少到0,也就導致了切換場景的dealloc方法沒有被調用,這也就引起了內存泄漏。

NSTimer

NSTimer會造成循環引用,timer會強引用target即self,在加入runloop的操作中,又引用了timer,所以在timer被invalidate之前,self也就不會被釋放。
所以我們要注意,不僅僅是把timer當作實例變量的時候會造成循環引用,只要申請了timer,加入了runloop,并且target是self,雖然不是循環引用,但是self卻沒有釋放的時機。如下方式申請的定時器,self已經無法釋放了。

NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(commentAnimation) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

解決這種問題有幾個實現方式,大家可以根據具體場景去選擇:

增加startTimer和stopTimer方法,在合適的時機去調用,比如可以在viewDidDisappear時stopTimer,或者由這個類的調用者去設置。

每次任務結束時使用dispatch_after方法做延時操作。注意使用weakself,否則也會強引用self。
- (void)startAnimation
{
    WS(weakSelf);
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [weakSelf commentAnimation];
    });
}

使用GCD的定時器,同樣注意使用weakself。

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

推薦閱讀更多精彩內容