Android|修复阿里云播放器下载不回调的问题

简介: 虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。

最近在升级 Android 项目里的阿里云播放器 SDK 版本,其中很多相关逻辑是基于阿里云提供的 Demo 来更新的。修改完自测时,发现下载器的回调接口偶现不回调的问题。本文简要记录解决过程。

问题描述

首先来看有问题的代码,Demo 里下载相关的有这么一段:

// AliyunDownloadManager.java

public void prepareDownload(final VidAuth vidAuth) {
    // some code here
    final AliMediaDownloader downloader = AliDownloaderFactory.create(mContext);
    downloader.setOnPrepareListener(new AliMediaDownloader.OnPreparedListener() {
            @Override
            public void onPrepared(MediaInfo mediaInfo) {
                // some code here
            }
    });
    setErrorListener(downloader, null);
    // some code here
}

private void setErrorListener(final AliMediaDownloader jniDownloader, final AliyunDownloadMediaInfo aliyunDownloadMediaInfo) {
    // some code here
    jniDownloader.setOnErrorListener(new AliMediaDownloader.OnErrorListener() {
        @Override
        public void onError(ErrorInfo errorInfo) {
            // some code here
        }
    });
}

调用这个方法后,正常情况下 onPrepared 和 onError 必被调用到其一,但现实是偶尔会出现两个都没被回调的情况。

那种感觉就像,你约好了朋友一起吃饭,结果他突然失联了,打电话不接,发消息不回,然后你就一个人在那里等着。这种事情在生活里不能忍,在代码里能忍?

排查和修改

通过一番艰苦的排查,久久没有思路,只好闷闷不乐地下班回家了。晚上洗澡的时候,突然想到,会不会是 downloader 被 GC 了?赶紧拿小本本记下思路,第二天一早翻出这块代码一看,果然如此。

downloader 是在方法内部创建的局部变量,方法执行完毕后,downloader 就会被释放,如果此时发生 GC,就会回收它对应的对象。到了回调时机,发现 downloader 对应的对象已经被回收了,回调也就无从谈起了

那要打破这个局面,就需要将 downloader 的引用保持住,在必要的回调发生以后再释放。比如我们可以想到,这段代码所在类是一个单例,那么在它里面声明一个 List,将 downloader 放进去,等回调结束后再移除。这样就能保证 downloader 在回调发生时还没有被回收。

修改后的代码类似这样:

// AliyunDownloadManager.java
private List<AliMediaDownloader> mJniDownloadLists = new ArrayList<>();

public void prepareDownload(final VidAuth vidAuth) {
    // some code here
    final AliMediaDownloader downloader = AliDownloaderFactory.create(mContext);
    downloader.setOnPrepareListener(new AliMediaDownloader.OnPreparedListener() {
            @Override
            public void onPrepared(MediaInfo mediaInfo) {
                // some code here
                // 增加了这一行
                mJniDownloadLists.remove(downloader);
            }
    });
    setErrorListener(downloader, null);
    // some code here
    // 增加了这一行
    mJniDownloadLists.add(downloader);
}

private void setErrorListener(final AliMediaDownloader jniDownloader, final AliyunDownloadMediaInfo aliyunDownloadMediaInfo) {
    // some code here
    jniDownloader.setOnErrorListener(new AliMediaDownloader.OnErrorListener() {
        @Override
        public void onError(ErrorInfo errorInfo) {
            // some code here
            // 增加了这一行
            mJniDownloadLists.remove(jniDownloader);
        }
    });
}

在尝试做以上修改时,发现了一个 彩蛋

阿里云的官方 Demo 里,其实已经声明了 mJniDownloadLists,且有注释 保存Downloader防止循环创建时,导致内存不足被回收,无法回调的问题,并且在其它类型的 prepareDownload 里也做了类似的处理,但不知道为什么在 VidAuth 类型参数的 prepareDownload 里没有,可能是漏掉了。

小结

在 Android / Java 项目里,类似的场景应该并不少见。虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。


如果读完文章有收获,可以关注我的微信公众号「闷骚的程序员」并🌟设为星标🌟,随时阅读更多内容。

目录
相关文章
|
21天前
|
移动开发 JavaScript 应用服务中间件
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
145 5
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
27天前
|
移动开发 前端开发 Android开发
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
211 12
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
29天前
|
移动开发 Rust JavaScript
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
450 3
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
26天前
|
移动开发 Android开发
【03】建立隐私关于等相关页面和内容-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【03】建立隐私关于等相关页面和内容-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
92 0
|
4月前
|
存储 Android开发 数据安全/隐私保护
Thanox安卓系统增加工具下载,管理、阻止、限制后台每个APP运行情况
Thanox是一款Android系统管理工具,专注于权限、后台启动及运行管理。支持应用冻结、系统优化、UI自定义和模块管理,基于Xposed框架开发,安全可靠且开源免费,兼容Android 6.0及以上版本。
334 4
|
4月前
|
存储 人工智能 文字识别
三款安卓手机word编辑器下载,Microsoft Word,wps office,Word手机版,手机word编辑查看阅读器,PDF转换器apk下载
WPS Office是一款功能强大的办公软件,支持文档编辑、表格处理和演示文稿制作,兼容多种格式并提供丰富的云服务。它具备低内存占用、快速运行的特点,支持跨设备同步与多人协作,内置海量模板及AI辅助功能,如智能写作和PPT自动生成。此外,还可扫描文件、编辑PDF并转换为其他格式,极大提升办公效率,适合手机用户便捷操作。
365 1
|
8月前
|
JavaScript Linux 网络安全
Termux安卓终端美化与开发实战:从下载到插件优化,小白也能玩转Linux
Termux是一款安卓平台上的开源终端模拟器,支持apt包管理、SSH连接及Python/Node.js/C++开发环境搭建,被誉为“手机上的Linux系统”。其特点包括零ROOT权限、跨平台开发和强大扩展性。本文详细介绍其安装准备、基础与高级环境配置、必备插件推荐、常见问题解决方法以及延伸学习资源,帮助用户充分利用Termux进行开发与学习。适用于Android 7+设备,原创内容转载请注明来源。
1796 77
|
4月前
|
API 开发工具 Android开发
qq虚拟视频插件下载安装手机版, 安卓虚拟视频插件,替换摄像头工具
Xposed入口模块:拦截目标应用的相机调用‌23 Camera1 API处理:通过PreviewCallback替换视频流‌1 Camera2 API适
|
8月前
|
存储 编解码 开发工具
Android平台毫秒级低延迟HTTP-FLV直播播放器技术探究与实现
本文详细探讨了在Android平台上实现HTTP-FLV播放器的过程。首先介绍了FLV格式的基础,包括文件头和标签结构。接着分析了HTTP-FLV传输原理,通过分块传输实现流畅播放。然后重点讲解了播放器的实现步骤,涵盖网络请求、数据解析、音视频解码与渲染,以及播放控制功能的设计。文章还讨论了性能优化和网络异常处理的方法,并总结了HTTP-FLV播放器的技术价值,尤其是在特定场景下的应用意义。
339 11
|
XML 缓存 Android开发
Android MediaPlayer 音乐播放器扫描 本地音乐、上一曲、下一曲切歌、播放本地音乐(下)
Android MediaPlayer 音乐播放器扫描 本地音乐、上一曲、下一曲切歌、播放本地音乐(下)
337 0
Android MediaPlayer 音乐播放器扫描 本地音乐、上一曲、下一曲切歌、播放本地音乐(下)

热门文章

最新文章