2

我收到了来自用户的大量错误报告,这些错误报告来自强制关闭对话框,其中陈述如下:

java.lang.ExceptionInInitializerError
at com.MyRequest.execute(MyRequest.java:59)
at org.nsdev.MySyncAdapter.onPerformSync(MySyncAdapter.java:90)
at android.content.AbstractThreadedSyncAdapter$SyncThread.run(AbstractThreadedSyncAdapter.java:164)
  Caused by: java.lang.RuntimeException: 
  Can't create handler inside thread that has not called Looper.prepare()
at android.os.Handler.<init>(Handler.java:121)
at android.os.AsyncTask$InternalHandler.<init>(AsyncTask.java:421)
at android.os.AsyncTask$InternalHandler.<init>(AsyncTask.java:421)
at android.os.AsyncTask.<clinit>(AsyncTask.java:152)

现在,我一直在试图弄清楚我应该如何最好地解决这个问题。在我的 onPerformSync 实现中,我创建了许多 AsyncTask 并执行它们。我怀疑我的问题是在从 onPerformSync 返回之前我没有等待这些任务完成。我尝试执行以下操作:

Looper.prepare();

// Execute async tasks

Looper.loop();

然后我设置了一个计数器,当该计数器减为零时,在异步任务的回调中调用 Looper.myLooper().quit()。

但是这样做会导致我的应用程序因以下运行时错误而更加崩溃:

java.lang.RuntimeException: Main thread not allowed to quit
    at android.os.MessageQueue.enqueueMessage(MessageQueue.java:175)
    at android.os.Looper.quit(Looper.java:173)
    at org.nsdev.MySyncAdapter$1.requestFinished(MySyncAdapter.java:89)
    at com.MyAsyncTask.onPostExecute(MyAsyncTask.java:84)
    at android.os.AsyncTask.finish(AsyncTask.java:417)
    at android.os.AsyncTask.access$300(AsyncTask.java:127)
    at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:429)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:123)
    at android.app.ActivityThread.main(ActivityThread.java:4627)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:521)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:868)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:626)
    at dalvik.system.NativeStart.main(Native Method)

知道如何正确完成我的同步吗?

4

3 回答 3

4

我知道这篇文章已经过时了,但是对于遇到这个帖子的其他人来说,在主线程上不调用 looper.quit() 怎么样?以下从产生几个 AsyncTasks 的 SyncAdpater 服务对我有用,尽管在性能方面可能有其他原因不这样做......

Handler h = new Handler(Looper.getMainLooper());
            h.post(new Runnable() {
                public void run() {

                    // AsyncTask... execute();
                }
            });
于 2012-09-30T17:35:35.673 回答
2

其实你不能。AsyncTask 使用主 UI 线程并依赖于 Handler 和 Looper,这在其中不可用。使用纯线程实现,AsyncTask 应该只在活动中使用。

于 2012-05-23T17:53:41.580 回答
0

aloc 的响应解决了这个问题,但是这个解决方案的问题是 UI 冻结,这个问题存在于 Android 2.2 和 Android 2.3 中。我发现的技巧是在我的类中调用 AsyncTask 的构造函数SyncAdapter并使用Looper.prepare()

/**
 * Constructor. Obtains handle to content resolver for later use.
 */
public SyncAdapter(Context context, boolean autoInitialize) {
    super(context, autoInitialize);
    mHandler = new Handler();
    //Crear un dummy AsyncTask
    new AsyncTask<Void, Void,String>() {
        @Override
        protected String doInBackground(Void... params) {
            return null;
        }
    }.execute(null,null);
}
于 2014-01-09T13:57:01.210 回答