我们有一个内部应用程序,可在启动时检查更新。由于它仅供内部使用,因此未在Play商店中发布。
以下更新代码在我的测试设备(API 23)上正常运行。但是,我最近发布了一个更新,该更新可在测试设备上下载,但报告:
解析错误-解析软件包时出现问题
在我自己的设备(API 24)上,应用下载并崩溃。
public class UpdateApp extends AsyncTask<String,Void,Void> {
private Context context;
public void setContext(Context contextf){
context = contextf;
}
@Override
protected Void doInBackground(String... arg0) {
try {
URL url = new URL(arg0[0]);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setDoOutput(true);
conn.connect();
File file = context.getCacheDir();
file.mkdirs();
outputFile = new File(file, "update.apk");
if(outputFile.exists()){
outputFile.delete();
}
FileOutputStream fos = new FileOutputStream(outputFile);
InputStream is = conn.getInputStream();
byte[] buffer = new byte[1024];
int len1 = 0;
while ((len1 = is.read(buffer)) != -1) {
fos.write(buffer, 0, len1);
}
fos.close();
is.close();
outputFile.setReadable(true, false);
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(outputFile), "application/vnd.android.package-archive");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
} catch (Throwable ex) {
Toast.makeText(context, ex.toString(), Toast.LENGTH_LONG).show();
}
return null;
}
在这两种设备上,在抛出错误(API 23)或崩溃(API 24)之前,都已下载更新的APK。我不明白该应用程序如何崩溃,因为所有代码都位于try
类的catch
块的Throwable
块中,因此它应该catch
所有错误和例外!
以下是我的API 24设备的logcat
输出,同时在下载文件后进行更新。除了指出AysncTask
类存在的问题之外,我无法从中得出任何有用的信息。
2018-11-04 16:47:27.593 966-1017/? E/PROXIMITY: ProximitySensor: unknown event (type=3, code=0)
2018-11-04 16:47:27.704 23431-23476/uk.co.letsdelight.letsdelight E/AndroidRuntime: FATAL EXCEPTION: AsyncTask #1
Process: uk.co.letsdelight.letsdelight, PID: 23431
java.lang.RuntimeException: An error occurred while executing doInBackground()
at android.os.AsyncTask$3.done(AsyncTask.java:318)
at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:354)
at java.util.concurrent.FutureTask.setException(FutureTask.java:223)
at java.util.concurrent.FutureTask.run(FutureTask.java:242)
at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:243)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607)
at java.lang.Thread.run(Thread.java:761)
Caused by: java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()
at android.os.Handler.<init>(Handler.java:208)
at android.os.Handler.<init>(Handler.java:122)
at android.widget.Toast$TN.<init>(Toast.java:351)
at android.widget.Toast.<init>(Toast.java:106)
at android.widget.Toast.makeText(Toast.java:265)
at uk.co.letsdelight.letsdelight.Update$UpdateApp.doInBackground(Update.java:123)
at uk.co.letsdelight.letsdelight.Update$UpdateApp.doInBackground(Update.java:83)
at android.os.AsyncTask$2.call(AsyncTask.java:304)
at java.util.concurrent.FutureTask.run(FutureTask.java:237)
at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:243)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1133)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:607)
at java.lang.Thread.run(Thread.java:761)
2018-11-04 16:47:27.716 966-989/? W/ActivityManager: Force finishing activity uk.co.letsdelight.letsdelight/.MainActivity
2018-11-04 16:47:27.745 966-23481/? W/AES: Exception Log handling...
我还可以做些什么来尝试解决此问题,或者使代码更健壮,以使我们在进行进一步更新时不会遇到问题?
答案 0 :(得分:1)
步骤1:将Toast
换成其他东西。您无法在后台线程上显示Toast
,这是错误消息告诉您的内容(Can't create handler inside thread that has not called Looper.prepare()
)。考虑到您仍在使用AsyncTask
,一种典型的解决方案是在Exception
的字段中保留AsyncTask
,然后在onPostExecute()
中对其进行操作,如下所示在主应用程序线程上执行。
步骤2:始终记录异常。至少,您应该登录到LogCat。如果您已经集成了ACRA,Crashlytics或类似的解决方案,则应采取步骤确保您的异常也得到报告。现在,您的catch
块仅 试图显示Toast
,这意味着您无法知道具体问题是什么(例如,缺乏连通性)。
步骤3:Android 7.0+上的一个特定问题是您不能很好地使用Uri.fromFile()
,因为file
Uri
的值被禁止了。在这些设备上,使用FileProvider
使安装程序可以使用APK,并使用FileProvider.getUriForFile()
将Uri
放入您的Intent
中。不幸的是,在Android 6.0及更低版本的设备上,您不能使用FileProvider
,因为安装程序将不支持content
Uri
值。因此,您将需要检测您的版本(Build.VERSION.SDK_INT
),并根据版本使用正确的Uri
。
答案 1 :(得分:0)
有时java.lang.RuntimeException: Can't create handler inside thread that has not called Looper.prepare()
发生在AsyncTasks中,因为您在后台线程中调用execute
。
您可以附加调试器,并检查是否在后台执行此asyncTask。您可以使用Evaluator来获取Thread.currentThread().getId()
的值(如果它返回1),则表明您在主线程中,否则请尝试在应用程序的主线程中调用execute