iOS NSTimer循環(huán)引用的幾種解決辦法
發(fā)生場(chǎng)景
在 Controller B 中有一個(gè) NSTimer
@property (strong, nonatomic) NSTimer *timer;
你創(chuàng)建了它,并掛載到 main runloop
self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(timerAction:) userInfo:nil repeats:true];
然后退出 Controller B 的時(shí)候,忘記關(guān)掉 timer 了
Controller B 將不會(huì)釋放,B 與 timer 循環(huán)引用。因?yàn)閯?chuàng)建 timer 的時(shí)候把 self 直接寫(xiě)進(jìn)去了。
方法一
既然不能直接傳 self,那傳 weakSelf 試試
__weak typeof(self) weakSelf = self; self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:weakSelf selector:@selector(timerAction:) userInfo:nil repeats:true];
測(cè)試結(jié)果還是發(fā)生了循環(huán)引用,B 沒(méi)有釋放,timer 對(duì) weakSelf 這個(gè)變量是強(qiáng)引用的,timer -> weakSelf -> B -> timer,三者之間形成循環(huán)引用。
方法二
設(shè)置一個(gè)包裝類(lèi),包著 Controller B 放進(jìn) timer 中,像這樣

我認(rèn)為 Controller B 有幾 MB 那么大,泄露了很浪費(fèi)內(nèi)存。
WeakWrap 只有幾百個(gè)字節(jié)那么小,泄露了也沒(méi)關(guān)系。
WeakWrap 中對(duì) Controller B 弱引用,WeakWrap 包著 Controller B,傳進(jìn) timer 中,就算忘記關(guān) timer,也只是泄露了 WeakWrap 和 timer。
理論上還是有內(nèi)存泄露,只不過(guò)比較少,如果一個(gè) Controller 是頻繁進(jìn)出的,進(jìn)出一次,丟失一個(gè),如果有幾十個(gè)泄露的 timer 掛在 main runloop 上會(huì)影響性能和流暢性,你想幾十個(gè) timer 一起 fire,又調(diào)用了 timer 事件響應(yīng)方法,開(kāi)銷(xiāo)還是挺大的。
方法三
NSTimer 已知是會(huì)強(qiáng)引用參數(shù) target:self 的了,如果忘記關(guān) timer 的話(huà),傳什么進(jìn)去都會(huì)被強(qiáng)引用。干脆實(shí)現(xiàn)一個(gè) timer 算了,timer 的功能就是定時(shí)調(diào)某個(gè)方法,NSTimer 的調(diào)用時(shí)間是不精確的!它掛在 runloop 上受線(xiàn)程切換,上一個(gè)事件執(zhí)行時(shí)間的影響。
利用 dispatch_asyn() 定時(shí)執(zhí)行函數(shù)??聪旅娲a。
- (void)loop {
[self doSomething];
......
// 休息 time 秒,再調(diào) loop,實(shí)現(xiàn)定時(shí)調(diào)用
[NSThread sleepForTimeInterval:time];
dispatch_async(self.runQueue, ^{
[weakSelf loop];
});
}
dispatch_async 中調(diào) loop 不會(huì)產(chǎn)生遞歸調(diào)用
dispatch_async 是在隊(duì)列中添加一個(gè)任務(wù),由 GCD 去回調(diào) [weakSelf loop]
這辦法解決了timer 不能釋放,掛在 runloop 不能移除的問(wèn)題。
利用這方法,我寫(xiě)了個(gè)不會(huì)發(fā)生循環(huán)引用的 timer,controller 釋放,timer 也自動(dòng)停止釋放,甚至 timer 的 block 里面可以直接寫(xiě) self,也不會(huì)循環(huán)引用。github下載地址
方法四
NSTimer 我之前沒(méi)遇到過(guò)循環(huán)引用的問(wèn)題,因?yàn)槲乙恢倍际桥鋵?duì)使用,在 viewWillAppear 開(kāi)啟,在 viewWillDisappear 關(guān)閉,不關(guān)閉的話(huà)那么多 timer 掛載在 runloop 上感覺(jué)挺影響性能和流暢性的,就像管理內(nèi)存一樣,申請(qǐng)和釋放配對(duì)使用,就不會(huì)泄露了,誰(shuí)申請(qǐng)誰(shuí)釋放的原則。但是很大的團(tuán)隊(duì)的話(huà),別人可能會(huì)寫(xiě)錯(cuò),造成泄露,可以從技術(shù)上,團(tuán)隊(duì)編程規(guī)范上解決他。
比如定一些規(guī)范,Controller 退出一定要成功銷(xiāo)毀,不能泄露內(nèi)存。Block 里不能寫(xiě) self 等等。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
IOS UI學(xué)習(xí)教程之設(shè)置UITextField各種屬性
這篇文章主要為大家詳細(xì)介紹了IOS UI學(xué)習(xí)教程之設(shè)置UITextField各種屬性,感興趣的小伙伴們可以參考一下2016-03-03
iOS使用CoreMotion實(shí)現(xiàn)搖一搖功能
這篇文章主要為大家詳細(xì)介紹了iOS使用CoreMotion實(shí)現(xiàn)搖一搖功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06
iOS開(kāi)發(fā)中使用CoreLocation框架處理地理編碼的方法
這篇文章主要介紹了iOS開(kāi)發(fā)中使用CoreLocation框架處理地理編碼的方法,代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-12-12
iOS 水波紋動(dòng)畫(huà)的實(shí)現(xiàn)效果
本篇文章主要介紹了iOS 水波紋的實(shí)現(xiàn)的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01
iOS schem與Universal Link 調(diào)試時(shí)踩坑解決記錄
這篇文章主要為大家介紹了iOS schem與Universal Link 調(diào)試時(shí)踩坑解決記錄,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
iOS點(diǎn)擊查看大圖的動(dòng)畫(huà)效果
這篇文章主要為大家詳細(xì)介紹了iOS點(diǎn)擊查看大圖的動(dòng)畫(huà)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
iOS自定義UIBarButtonItem的target和action示例代碼
這篇文章主要給大家介紹了關(guān)于iOS自定義UIBarButtonItem的target和action的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
IOS開(kāi)發(fā)之路--C語(yǔ)言存儲(chǔ)方式和作用域
只有你完全了解每個(gè)變量或函數(shù)存儲(chǔ)方式、作用范圍和銷(xiāo)毀時(shí)間才可能正確的使用這門(mén)語(yǔ)言。今天將著重介紹C語(yǔ)言中變量作用范圍、存儲(chǔ)方式、生命周期、作用域和可訪(fǎng)問(wèn)性。2014-08-08

