- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我有一个 AsyncTask
获取一些数据,然后用这些新数据更新 UI。几个月来它一直运行良好,但我最近添加了一个功能,当有新数据时显示通知。现在,当我的应用程序通过通知启动时,有时我会收到此异常并且未调用 onPostExecute
。
这是应用启动时发生的情况:
1) 展开 UI 并找到 View
2) 取消检查新数据的警报(通过AlarmManager
)并重置警报。 (这样如果用户禁用警报,它会在他/她下次重新启动之前被取消。)
3) 启动AsyncTask
。如果应用是通过通知启动的,请传入一点数据,然后取消通知。
我对可能导致此异常的原因感到困惑。似乎异常来自 AsyncTask
代码,所以我不确定如何修复它。
谢谢!
这是一个异常(exception):
I/My App( 501): doInBackground exiting
W/MessageQueue( 501): Handler{442ba140} sending message to a Handler on a dead thread
W/MessageQueue( 501): java.lang.RuntimeException: Handler{442ba140} sending message to a Handler on a dead thread
W/MessageQueue( 501): at android.os.MessageQueue.enqueueMessage(MessageQueue.java:179)
W/MessageQueue( 501): at android.os.Handler.sendMessageAtTime(Handler.java:457)
W/MessageQueue( 501): at android.os.Handler.sendMessageDelayed(Handler.java:430)
W/MessageQueue( 501): at android.os.Handler.sendMessage(Handler.java:367)
W/MessageQueue( 501): at android.os.Message.sendToTarget(Message.java:348)
W/MessageQueue( 501): at android.os.AsyncTask$3.done(AsyncTask.java:214)
W/MessageQueue( 501): at java.util.concurrent.FutureTask$Sync.innerSet(FutureTask.java:252)
W/MessageQueue( 501): at java.util.concurrent.FutureTask.set(FutureTask.java:112)
W/MessageQueue( 501): at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:310)
W/MessageQueue( 501): at java.util.concurrent.FutureTask.run(FutureTask.java:137)
W/MessageQueue( 501): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1068)
W/MessageQueue( 501): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:561)
W/MessageQueue( 501): at java.lang.Thread.run(Thread.java:1096)
编辑:这是我的主要 Activity (由通知打开的 Activity )中的 onCreate
方法。为了节省空间,我省略了一些 onClickListeners
。我不认为它们应该有任何效果,因为它们所连接的按钮没有被按下。
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); // Call the parent
setContentView(R.layout.main); // Create the UI from the XML file
// Find the UI elements
controls = (SlidingDrawer) findViewById(R.id.drawer); // Contains the
// buttons
// comic = (ImageView) findViewById(R.id.comic); // Displays the comic
subtitle = (TextView) findViewById(R.id.subtitleTxt); // Textbox for the
// subtitle
prevBtn = (Button) findViewById(R.id.prevBtn); // The previous button
nextBtn = (Button) findViewById(R.id.nextBtn); // The next button
randomBtn = (Button) findViewById(R.id.randomBtn); // The random button
fetchBtn = (Button) findViewById(R.id.comicFetchBtn); // The go to specific id button
mostRecentBtn = (Button) findViewById(R.id.mostRecentBtn); // The button to go to the most recent comic
comicNumberEdtTxt = (EditText) findViewById(R.id.comicNumberEdtTxt); // The text box to Zooming image view setup
zoomControl = new DynamicZoomControl();
zoomListener = new LongPressZoomListener(this);
zoomListener.setZoomControl(zoomControl);
zoomComic = (ImageZoomView) findViewById(R.id.zoomComic);
zoomComic.setZoomState(zoomControl.getZoomState());
zoomComic.setImage(BitmapFactory.decodeResource(getResources(), R.drawable.defaultlogo));
zoomComic.setOnTouchListener(zoomListener);
zoomControl.setAspectQuotient(zoomComic.getAspectQuotient());
resetZoomState();
// enter the new id
imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); // Used to hide the soft keyboard
Log.i(LOG_TAG, "beginning loading of first comic");
int notificationComicNumber = getIntent().getIntExtra("comic", -1);
Log.i(LOG_TAG, "comic number from intent: " + notificationComicNumber);
if (notificationComicNumber == -1) {
fetch = new MyFetcher(this, zoomComic, subtitle, controls, comicNumberEdtTxt, imm, zoomControl);
fetch.execute(MyFetcher.LAST_DISPLAYED_COMIC);
} else {
fetch = new MyFetcher(this, zoomComic, subtitle, controls, comicNumberEdtTxt, imm, zoomControl);
fetch.execute(notificationComicNumber);
((NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE)).cancelAll();
}
Log.i(LOG_TAG, "ending loading of new comic");
Log.i(LOG_TAG, "first run checks beginning");
// Get SharedPreferences
prefs = getSharedPreferences("prefs", Context.MODE_PRIVATE);
// Check if this is the first run of the app for this version
if (prefs.getBoolean("firstRun-" + MAJOR_VERSION_NUMBER, true)) {
prefs.edit().putBoolean("firstRun-" + MAJOR_VERSION_NUMBER, false).commit();
firstRunVersionDialog();
}
// Check if this is the first run of the app
if (prefs.getBoolean("firstRun", true)) {
prefs.edit().putBoolean("firstRun", false).commit();
firstRunDialog();
}
Log.i(LOG_TAG, "First run checks done");
// OnClickListener s for the buttons omitted to save space
编辑 2:我一直在挖掘 Android 源代码,以追踪异常的来源。这是 Handler
中 sendMessageAtTime
的第 456 和 457 行:
msg.target = this;
sent = queue.enqueueMessage(msg, uptimeMillis);
这是 MessageQueue
中的 enqueueMessage
:
final boolean enqueueMessage(Message msg, long when) {
if (msg.when != 0) {
throw new AndroidRuntimeException(msg
+ " This message is already in use.");
}
if (msg.target == null && !mQuitAllowed) {
throw new RuntimeException("Main thread not allowed to quit");
}
synchronized (this) {
if (mQuiting) {
RuntimeException e = new RuntimeException(
msg.target + " sending message to a Handler on a dead thread");
Log.w("MessageQueue", e.getMessage(), e);
return false;
} else if (msg.target == null) {
mQuiting = true;
}
msg.when = when;
//Log.d("MessageQueue", "Enqueing: " + msg);
Message p = mMessages;
if (p == null || when == 0 || when < p.when) {
msg.next = p;
mMessages = msg;
this.notify();
} else {
Message prev = null;
while (p != null && p.when <= when) {
prev = p;
p = p.next;
}
msg.next = prev.next;
prev.next = msg;
this.notify();
}
}
return true;
}
我对 mQuiting
是什么有点困惑,但看起来上一次 enqueueMessage
被称为 msg.target
为 null .
最佳答案
这是由于 Android 框架中的 AsyncTask 中的错误造成的。 AsyncTask.java 代码如下:
private static final InternalHandler sHandler = new InternalHandler();
它希望在主线程上初始化它,但这不能保证,因为它会在恰好导致类运行其静态初始化程序的任何线程上初始化。我在 Handler 引用工作线程时重现了这个问题。
导致这种情况发生的常见模式是使用 IntentService 类。 C2DM 示例代码执行此操作。
一个简单的解决方法是将以下代码添加到应用程序的 onCreate 方法中:
Class.forName("android.os.AsyncTask");
这将强制 AsyncTask 在主线程中初始化。我在 android 错误数据库中提交了一个错误。见 http://code.google.com/p/android/issues/detail?id=20915 .
关于android - onPostExecute 未在 AsyncTask 中调用(处理程序运行时异常),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4280330/
当我使用 new GiveMeJson2.execute(ref_id).get() 调用异步类时, 我的应用程序进入 doInBackground,创建一个 JSON 字符串并返回它。 此时,我认为
我正在尝试在后台下载一些文件,为此我使用了下载管理器并制作了此方法: /*** media download ***/ public long mediaDownload(ArrayList
我在尝试使用 onPostExecute 来触发这段代码时遇到问题,我对 java 尤其是多线程很陌生,所以如果我正在做的事情完全错误,请随时纠正我...需要学习. 所以问题是这样的: public
以下代码是我出于 Intent 而添加的代码,结果是从我的login.php 返回的内容。但它无法比较成功与否。在任何情况下它都不会进入 if 循环。但它在警报对话框中显示“登录成功”。 @Overr
字符串 descript 的内容显示在 logcat 中,没有任何问题,但不会显示在 OnPostExecute 对话框中的 TextView 上。这是一个空指针异常。我不知道为什么,因为 episo
当我运行我的应用程序时,onPostExecute似乎没有被调用。它没有像应有的那样填充到 UI 中。另外,关于DoInBackground ,任何超过 for 循环的日志消息: for (int i
我有一个单独的类(不是 Activity)来管理图像及其属性。 在这个类(Image.class)中,我还创建了一个 AsynkTask 来从网络下载位图。一切正常,除了 AsyncTask 继续运行
在 php 脚本的支持下,我有一段代码在 Async Task 中运行。我只想检查我的数据库中是否存在电子邮件 ID。我想显示 toast :“电子邮件已存在”(如果在数据库中)。否则另一个 toas
所以我创建了一个 web 服务调用类,它调用我在 AsyncTask 上扩展的 web 服务: public class WebServiceLoginCall extends AsyncTask{
我有一个扩展 AsyncTask 的类,它旨在用作我的应用程序的通用任务管理器类。 问题:奇怪的行为是进度对话框出现,但从未被关闭。 我确信每个任务实例都会调用 onPostExecute(),因为任
更新,请见下文。 如何将我的类 LocationData 和我的 ArrayList listOfObjects 返回到 onPostExecute()?我想在我的 UI 中使用它,现在它在 Asyn
我正在尝试获取 onPostExecute()完成 doInBackground() 后调用的方法方法,但它没有被调用。我该如何解决这个问题? 我已经调试过了,可以看到ArrayList data里面
我有一些控制权,因此我做了一些线程。我想用 asycntask 将一个屏幕的加载栏显示到另一个屏幕。我的问题是加载栏正在快速显示和消失,然后在没有加载的情况下等待同一个屏幕很多bar.And 一段时间
我在 Android 应用程序上做一个登录系统,当我输入错误的帐户时,onPostExecute 方法返回正确的失败消息,但是当我输入一个现有帐户并且登录正确后它仍然是相同的失败消息,我不明白为什么!
public static class TestTask extends AsyncTask { private String stacktrace; public TestTask
我有一个运行良好的 AsyncTask 类,但在这个类中,我在 onPreExecute 中有一个 ProgressDialog,它工作正常,但我需要在完成进程对话框关闭时,我将代码放在 onPost
我不知道如何找出这个空指针异常发生的位置。 按照 Google Play 的建议,崩溃只在用户的物理设备上发生过一次(到目前为止)——我无法在调试环境中重现它。 Google Play 给我的 Sta
我尝试使用此函数返回对象数组: public static JSONEvent[] invokeFunction(String funName, String requestContent) {
我有一个 Activity (RecipiesActivity),通过单击我的应用程序主 Activity 的“下一步”按钮打开。 在 RecipiesActivity onCreate 中,我想在
我有一个看起来像这样的类: internal class MyClass : AsyncTask>>> { protected override void OnPostExec
我是一名优秀的程序员,十分优秀!