显示对话框时出现BadTokenException

时间:2012-10-02 15:52:18

标签: android android-activity dialog alertdialog

我在Android应用程序上有一个活动,它启动与我的服务器的同步过程。这个过程消耗大量内存和处理,并且需要一些时间才能完成。

流程完成后,会向用户显示AlertDialog,显示流程是否已成功完成。

如果我离开活动前景,一切都会按预期工作。但是,有时候,当我离开应用程序时,后台进程继续它的工作,但是,当我返回应用程序时,它崩溃并出现以下错误:

android.view.WindowManager $ BadTokenException:无法添加窗口 - 令牌android.os.BinderProxy@4086ea48无效;你的活动在运行吗?

是否可能导致此错误,因为操作系统已销毁该活动?如果是,那么进程(在另一个线程上运行)如何继续运行?而且,如果我们确认问题是由活动的破坏引起的,我怎么能处理这种情况,避免崩溃并向用户显示对话框?

这是一些代码......

public class ActSincronizacao extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);
        setContentView(R.layout.sincronizacao);

        //Prepare so starts the process
        [...]

        // Defines a Handler that will be called when the process gets finished
        oSocketEngine.addProgressUpdateFinishListener(new ProgressUpdateFinishListener() {

            @Override
            public void ProgressUpdateFinishOccurred(final ProgressUpdateFinish evt) {

                //Do some processing that doesn´t envolve UI
                [...]

                // Process that envolves UI
                runOnUiThread(new Runnable() {

                    public void run() {

                        if ((boolean)evt.getArgs().getResult()) {

                            //If the process gets Success
                            AlertDialog.Builder builder = new AlertDialog.Builder(ActSincronizacao.this);
                            builder.setIcon(android.R.drawable.ic_dialog_info);
                            builder.setTitle("Informação");
                            builder.setMessage("Sincronização concluida com sucesso.");

                            // Defines a button handler
                            builder.setNeutralButton("OK", new OnClickListener() {

                                @Override
                                public void onClick(DialogInterface dialog, int which) {

                                    //Go to another Activity
                                    [...]


                                }
                            });

                            //Call the Dialog
                            builder.create().show();

                        } else {


                            //If something wrong hapenned

                            StringBuilder stb = new StringBuilder();
                            stb.append("Some error ocurrend while running the sync process.");

                            if (evt.getArgs().getMessage() != null) {
                                stb.append("\n\n");
                                stb.append(evt.getArgs().getMessage());
                            } else {
                                stb.append("\n\n");
                                stb.append("Try again...");
                            }

                            AlertDialog.Builder builder = new AlertDialog.Builder(ActSincronizacao.this);
                            builder.setIcon(android.R.drawable.ic_dialog_alert);
                            builder.setTitle("Error");
                            builder.setMessage(stb.toString());
                            builder.setNeutralButton("OK", null);
                            builder.create().show();            //<<<<<<<<<<<<<<<<< Crash is reported here

                        }
                    }
                });
            }
        });

        //Starts the process in backgroud (Thread)
        oSocketEngine.StartCommunication(oDeviceConfig);

    }

}

这是记录的完整堆栈跟踪:

android.view.WindowManager$BadTokenException: Unable to add window -- token android.os.BinderProxy@4086ea48 is not valid; is your activity running?
at android.view.ViewRoot.setView(ViewRoot.java:532)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:200)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:114)
at android.view.Window$LocalWindowManager.addView(Window.java:424)
at android.app.Dialog.show(Dialog.java:241)
at PortalExecutivoSales.Android.ActSincronizacao$3$1.run(ActSincronizacao.java:138)
at android.os.Handler.handleCallback(Handler.java:587)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:130)
at android.app.ActivityThread.main(ActivityThread.java:3687)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:507)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:867)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:625)
at dalvik.system.NativeStart.main(Native Method)

1 个答案:

答案 0 :(得分:5)

对于仍在寻找此问题解决方案的人:

以下链接说明了此问题发生的原因和原因。 http://vinnysoft.blogspot.com.br/2010/11/androidviewwindowmanagerbadtokenexcepti.html

我为解决这个问题做了什么:

1)。在ActSincronizacao类上创建了一个名为isActivityRunning的布尔变量,该变量将用于通知活动是否正在运行。此变量在onPause和onResume事件中设置,如下所示:

@Override
protected void onPause() {
    isActivityActive = false;
    super.onPause();
}

@Override
protected void onResume() {
    isActivityActive = true;
    super.onResume();
}

2)。当进程结束时,在调用Dialog之前,我会检查isActivityActive变量的状态。如果true,我会像往常一样打电话给对话。如果没有,我会创建一个通知,如下所示:

http://developer.android.com/guide/topics/ui/notifiers/notifications.html#NotificationResponse

3)。当用户点击通知时,我仍然希望向用户显示带有错误消息的警报(如果有)。因此,通知应将 ErrorMessage 传递给正在调用的活动。这可以通过向extra添加Notification Intent来完成。查看下面的示例,查看如何通过(@Vidar Vestnes'问题)和如何消费(@ pinaise的答案)通过通知创建的额外

How to send parameters from a notification-click to an activity?

4)。最后,当用户点击通知时,我想启动应用程序,如果它没有运行。如果它正在运行,我想把它带回前台。 默认行为是,当我们点击通知时,会创建一个传递给通知意图的活动的新实例。要避免创建新活动并将应用程序置于前台,请参阅下面的@santhosh响应:

re-open background application via notification item

希望将来能帮助其他人。

问候。