VideoView中遺留的問題
- VideoView是直接繼承SurfaceView
- VideoView中的openVideo可能會(huì)ANR
- VideoView中的release计露,stopPlayBack都會(huì)導(dǎo)致ANR蹦魔,因?yàn)檫@些方法都是同步執(zhí)行的湿诊,并且通過IPC服務(wù)交給MediaServer去釋放資源
SurfaceView的問題
- SurfaceView不會(huì)添加到View樹上构哺,并且顯示在所有View之上
- 在按Home鍵的時(shí)候,會(huì)讓Surface銷毀,并且在重新進(jìn)入APP的時(shí)候衰猛,讓Surface重建,在Surface重建的時(shí)候刹孔,SurfaceView那一塊是透明的啡省,顯示的會(huì)是Activity的背景
- 在上下滑動(dòng)的時(shí)候,會(huì)導(dǎo)致Surface繪制不及時(shí)髓霞,會(huì)有殘留
- 多個(gè)VideoView同時(shí)播放的時(shí)候卦睹,在SurfaceFlinger支持不好的手機(jī)上,會(huì)出現(xiàn)下一個(gè)SurfaceView的某一幀會(huì)顯示在上一個(gè)SurfaceView上
解決辦法
使用TextureView替換SurfaceView實(shí)現(xiàn)VideoView方库,因?yàn)門extureView是直接繼承View的分预,并且在ListView中滑動(dòng)的時(shí)候,也不會(huì)在滑動(dòng)的時(shí)候薪捍,有殘留(看起來像是普通的View繪制和SurfaceView的繪制是兩套)
ANR的問題
由于MediaPlayer中的release,reset配喳,stopPlayBack都是同步的酪穿。而我們?cè)贚istView的每個(gè)卡片中,都有可能出現(xiàn)視頻晴裹,而且需要自動(dòng)播放被济,由于沒有時(shí)間做視頻先下載,再播放涧团,所以選擇的是Android自帶的流播放只磷。會(huì)一段一段的將視頻先讀到緩沖區(qū),再播放泌绣。
而且當(dāng)視頻卡片在滑出屏幕之后钮追,需要把視頻暫停,在不可見的時(shí)候不進(jìn)行播放阿迈。節(jié)省系統(tǒng)資源元媚,并且節(jié)省用戶流量。而如果同時(shí)出現(xiàn)多個(gè)視頻的時(shí)候苗沧,會(huì)頻繁調(diào)用到上述導(dǎo)致ANR的方法刊棕,會(huì)很容易出現(xiàn)ANR
解決方案
問題1.在視頻劃出ListView的時(shí)候,停止播放視頻待逞。
解決方案:在ListView中調(diào)用setRecycleListener甥角,設(shè)置View回收的監(jiān)聽,因?yàn)長(zhǎng)istView的重用性识樱,會(huì)在View回收到scrap區(qū)的時(shí)候嗤无,通過這個(gè)Listener進(jìn)行一些處理震束,所以在這里根據(jù)View.getTag,找到視頻View的引用翁巍,調(diào)用stopPlayBack停止
問題2.頻繁調(diào)用release等方法導(dǎo)致ANR
解決方案:在視頻調(diào)用的時(shí)候驴一,建立一個(gè)釋放視頻資源的守護(hù)線程。在Android中灶壶,直接可以用HandlerThread肝断,因?yàn)檫@樣可以盡可能的讓資源的消耗達(dá)到最少,HandlerThread在沒有新事件到來的時(shí)候驰凛,都是處于wait狀態(tài)胸懈,直到有新事件的到來,才會(huì)被notify恰响,處理新事件趣钱。它里面也是通過一個(gè)Thread,在這個(gè)Thread中新建一個(gè)Looper胚宦,在Looper中沒有事件的話首有,則wait,一旦通過Handler發(fā)送新事件的話枢劝,則會(huì)被notify井联。所以會(huì)在子線程中加入一個(gè)隊(duì)列,當(dāng)需要release的MediaPlayer您旁,直接丟到子線程去進(jìn)行資源釋放烙常。
但是這樣會(huì)導(dǎo)致一個(gè)問題,就是Android維護(hù)的MediaPlayer的狀態(tài)機(jī)中的狀態(tài)可能會(huì)亂鹤盒,這時(shí)候就會(huì)拋出IllegalStateException蚕脏,目前對(duì)于這種異常,我們選擇了捕獲它侦锯。
MediaPlayer.setSurface
由于MediaPlayer.setSurface需要傳遞一個(gè)Surface驼鞭,然后再在這個(gè)Surface上進(jìn)行繪制,如果頻繁new Surface傳入的話尺碰,就會(huì)導(dǎo)致GrafficBuffer分配Surface失敗终议,從而MediaPlayer會(huì)回調(diào)onError中,顯示視頻不能播放葱蝗。所以盡可能一個(gè)視頻View用一個(gè)Surface穴张。