MediaPlayer的prepareAsync不回调onPrepared问题
一、问题排查
debug调试的framework日志信息:
根据日志,进入MediaPlayer的framework层看源码
STEP 1
先从
V/MediaPlayerNative: constructor
V/MediaPlayerNative: setListener
https://android.googlesource.com/platform/frameworks/av/+/master/media/libmedia/mediaplayer.cpp
可以看到这是正常的new了一个C++层的MediaPlayer
以及设置了一个Java层的弱引用给C++层的MediaPlayer监听,目前一切正常
STEP 2
再看后面的日志:
V/MediaPlayerNative: reset
......
V/MediaPlayerNative: setVideoSurfaceTexture
V/MediaPlayerNative: prepareAsync
可以看到这里对MediaPlayer的设置一切正常,没有异常。
包括后面通过url取获取HTTP流,以及获取大小都正常。
STEP 3
再看后面的日志:
V/PlayerBase: baseRelease() piid=679 state=1
......
发现MediaPlayer的父类PlayerBase居然被release了???
STEP 4
可以看到,这是MediaPlayer被系统给GC了。
V/MediaPlayerNative: setListener
disconnect
可以看到,这里是最后C++层的MediaPlayer也做了release
二、原因分析
通过debug调试日志,可以分析出问题出在MediaPlayer被JVM的GC调用了finalize。
三、解决办法
方案一
在调用MediaPlayer的prepareAsync后,开启一个异步延时线程,判断是否prepare成功,如果失败,则将MediaPlayer重置,在设置dataSource后再来一次prepare。
方案二
接入AOP框架,拦截MediaPlayer的finalize方法,在被GC系统调用前,记录状态改变。再做后续的重试处理。
考虑到AOP框架的大小,一般都上10M,如果考虑包体大小,可以采纳方案一,否则推荐方案二。
作者:DiffX —— 这个人在加班
出处:http://www.cnblogs.com/diffx/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步