我今天在我的 Android 项目中遇到了一个涉及 aSyncTasks 的问题,经过一些研究找到了答案,正如我与之交谈过的人都没有意识到的那样,所以我想我会与 SO 社区分享,以防有人发现它有任何用处。
我的问题的简要概述:
单击安装按钮时,我有一个 UI 活动类来下载多个文件的包。文件“捆绑包”最多分为四个单独的下载,并且“组”信息与文件的其他信息一起存储在自定义类中。单击下载按钮时,(最多)四个作业通过实用程序 DownloadUtilities 与 android DownloadManager 一起排队。
当 DownloadUtilities 将“组”下载排入队列时,DownloadManager 下载引用存储在“组”自定义类中以供以后使用。
Download Utilities 类具有一个 BroadcastReceiver 来确认文件并在组的每个元素完成下载时将下载引用更新为 0 以显示它们已完成。
组中的所有文件下载完成后,即可对其进行处理。这是由 DownloadReceiver 触发的 aSyncTask 完成的。
这反过来会触发广播以通知 UI 活动请求的下载已完成,以便可以相应地更新 UI。
到目前为止一切都很好,但是,当我尝试添加进度条以显示“组”的进度时,我发现了一个问题
为了更新 ProgressBar,我创建了一个新的 aSyncTask,它可以查询下载管理器以计算下载文件的总大小,然后每秒轮询一次以更新 ProgressBar 以显示当前的下载进度。
doInBackground() 线程由以下伪代码组成:
Check if file1 is being downloaded, if it is, query the download manager for the total file size and add it to total
Same for files2-4
While the download references are not all 0:
If file 1 is still downloading, get file1 download total
Same for file2-4
Update progress to sum of downloaded bytes/total
我发现的问题是系统会死锁。如果下载管理器告诉下载接收器文件已完成下载,并且 ProcessDownload aSyncTask onPreExecute() 正在运行,但 doInBackGround 从未运行,Logcat 会。
同时,下载监视器在 doInBackground() 中不断循环,因为 while 循环的条件从未被取消断言,因为取消断言是在 ProcessDownload doInBackground() 线程中完成的。
很明显 aSyncTask doInBackground() 方法是互斥的,因此会导致死锁,但我不知道为什么。过去,我从来没有故意以这样的方式构建我的代码,这会造成问题......
我的理解是 aSyncTasks 提供了一种开发人员友好的方式来多线程您的代码,但似乎情况并非如此......