ios 中的循環引用問題及解決


循環引用,指的是多個對象相互引用時,使得引用形成一個環形,導致外部無法真正是否掉這塊環形內存。其實有點類似死鎖。

舉個例子:A->B->C->....->X->B   ->表示強引用,這樣的B的引用計數就是2,假如A被系統釋放了,理論上A會自動減小A所引用的資源,就是B,那么這時候B的引用計數就變成了1,所有B無法被釋放,然而A已經被釋放了,所有B的內存部分就肯定無法再釋放再重新利用這部分內存空間了,導致內存泄漏。

情況一:delegate

Delegate是ios中開發中最常遇到的循環引用,一般在聲明delegate的時候都要使用弱引用weak或者assign

@property (nonatomic, weak, nullable) id <UITableViewDelegate> delegate;

當然怎么選擇使用assign還是weak,MRC的話只能用assign,在ARC的情況下最好使用weak,因為weak修飾的變量在是否后自動為指向nil,防止不安全的野指針存在

情況二:Block

Block也是比較常見的循環引用問題,在Block中使用了self容易出現循環引用,因此很多人在使用block的時候,加入里面有用到self的操作都會聲明一個__weak來修飾self。其實便不是這樣的,不是所有使用了Block都會出現Self循環引用問題,只有self擁有Block的強引用才會出現這種情況。

所以一般在函數中臨時使用Block是不會出現循環應用的,因為這時候Block引用是屬於棧的。當棧上的block釋放后,block中對self的引用計數也會減掉

當然不一定要Self對Block有直接的引用才會出現,假如self的變量B,B中有個Block變量,就容易出現這種情況,好的是在block出現循環引用的,xcode7會出現警告提示(之前版本不確定)。

情況三:NSTimer

這是一個神奇的NSTimer,當你創建使用NSTimer的時候,NSTimer會默認對當前self有個強引用,所有在self使用完成打算是否的時候,一定要先使用NSTimer的invalidate來停止是否時間控制對self的引用

[_timer invalidate];

 

----------------------------------凌亂的分割線-------------------------------------

上面說的是我們常見的,其實循環引用就是說我們的強引用形成了閉環,還會有很多自己寫的代碼中會出現,平時還是要注意寫法。當然xcode的instruments也能幫助到大家排除一些這樣類似的內存問題。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM