首先,请不要 tl;dr 这个。它只是很多可快速扫描的代码。实际的句子很少而且简洁 :)。
我有一个奇怪的问题。 ClassCastException
强制关闭了我的 Activity
,但该异常并未指出问题实际出在哪里。这是异常(exception)情况:
E/AndroidRuntime( 402): FATAL EXCEPTION: AsyncTask #4
E/AndroidRuntime( 402): java.lang.RuntimeException: An error occured while executing doInBackground()
E/AndroidRuntime( 402): at android.os.AsyncTask$3.done(AsyncTask.java:200)
E/AndroidRuntime( 402): at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:274)
E/AndroidRuntime( 402): at java.util.concurrent.FutureTask.setException(FutureTask.java:125)
E/AndroidRuntime( 402): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:308)
E/AndroidRuntime( 402): at java.util.concurrent.FutureTask.run(FutureTask.java:138)
E/AndroidRuntime( 402): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1088)
E/AndroidRuntime( 402): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:581)
E/AndroidRuntime( 402): at java.lang.Thread.run(Thread.java:1019)
E/AndroidRuntime( 402): Caused by: java.lang.ClassCastException: [Ljava.lang.Object;
E/AndroidRuntime( 402): at mypackage.DayActivity$EventsFetcher.doInBackground(DayActivity.java:1)
E/AndroidRuntime( 402): at android.os.AsyncTask$2.call(AsyncTask.java:185)
E/AndroidRuntime( 402): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:306)
E/AndroidRuntime( 402): ... 4 more
W/ActivityManager( 61): Force finishing activity mypackage/mypackage.DayActivity
如您所见,它说错误发生在 DayActivity.java
的第 1 行,这是默认的包声明(即 package mypackage
)。
这是我的doInBackground
方法。在这里,怪异加剧了,因为整个方法基本上是一个 try/catch
block :
@Override
protected List<Event> doInBackground(Void... params) {
try {
final List<Event> events;
if (connectivityReceiver.hasConnection()) {
events = WebApi.getEvents(Util.getSelectedUser(DayActivity.this).getId(), start, end);
}
else if (Util.isPeriodCached(DayActivity.this, start, end)) {
final List<Event> allEvents = (List<Event>) new Cache(DayActivity.this).get(Cache.EVENTS);
events = Util.filterEvents(allEvents, start, end, Util.getSelectedUser(DayActivity.this).getId());
}
else {
events = null;
}
return events;
}
catch (Exception e) {
Log.e(Const.TAG, "Could not fetch events for day view", e);
}
return null;
}
理论上,抛出的任何异常都应该被捕获并记录下来。我不明白这怎么可能发生。诀窍是在 onPostExecute
中,我运行这段代码:
if (!connectivityReceiver.hasConnection()) {
Util.retryDialog(DayActivity.this, R.string.no_cache, R.string.no_cache_msg,
new EventsFetcher(dialog), false).show();
}
这是retryDialog()
:
public static Dialog retryDialog(final Activity context, int titleRes, int messageRes,
final AsyncTask retry, final boolean finishOnCancel) {
return new AlertDialog.Builder(context)
.setTitle(titleRes)
.setMessage(messageRes)
.setPositiveButton(R.string.retry, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
if (retry != null) {
retry.execute();
}
}
})
.setNegativeButton(R.string.cancel, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
if (finishOnCancel) {
context.finish();
}
}
})
.create();
}
问题是任务运行一次,显示对话框,但是当我点击重试时抛出异常,导致我的应用程序崩溃。
最佳答案
您要传递什么 AsyncTask
来重试? AsyncTask
只能执行一次,因此您需要创建一个新实例,而不是尝试重新启动已执行的实例。取自javadoc for AsyncTask :
任务只能执行一次(第二次执行会抛出异常)
编辑:.. 这就是我解释堆栈跟踪的方式。在 AsyncTask 中执行 done 时出现问题。错误的地方是将 Object[]
转换为它不是的对象。可能在从 doInBackground
到 onPostExecute
时,但这只是一个大胆的猜测。可能是由于任务的重新启动引起的,如果这是您尝试执行的操作。
关于java - 为什么我的 AsyncTask 中会出现这个 ClassCastException?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6409960/