这个错误是什么,为什么会发生?

05-17 18:24:57.069: ERROR/WindowManager(18850): Activity com.mypkg.myP has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@44c46ff0 that was originally added here
05-17 18:24:57.069: ERROR/WindowManager(18850): android.view.WindowLeaked: Activity ccom.mypkg.myP has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@44c46ff0 that was originally added here
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.view.ViewRoot.<init>(ViewRoot.java:231)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:148)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:91)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.view.Window$LocalWindowManager.addView(Window.java:424)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.Dialog.show(Dialog.java:239)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at com.mypkg.myP$PreparePairingLinkageData.onPreExecute(viewP.java:183)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.os.AsyncTask.execute(AsyncTask.java:391)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at com.mypkg.myP.onCreate(viewP.java:94)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2544)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2621)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.ActivityThread.access$2200(ActivityThread.java:126)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1932)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.os.Handler.dispatchMessage(Handler.java:99)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.os.Looper.loop(Looper.java:123)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at android.app.ActivityThread.main(ActivityThread.java:4595)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at java.lang.reflect.Method.invokeNative(Native Method)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at java.lang.reflect.Method.invoke(Method.java:521)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:860)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618)
05-17 18:24:57.069: ERROR/WindowManager(18850):     at dalvik.system.NativeStart.main(Native Method)

当前回答

当您尝试在活动有效完成后显示警报时,会出现“活动已泄漏最初添加的窗口…”错误。

AFAIK,你有两个选择:

重新考虑登录alert:在实际退出activity之前调用对话框上的dismiss()。 将对话框放在不同的线程中,并在该线程上运行它(与当前活动无关)。

其他回答

我正在制作一个科学的应用程序,已经接近4.5万行,我使用异步任务,以便能够中断长任务,如果需要的话,用户点击一定的按钮。

因此,有一个响应式的用户界面,有时还有一个并行的长任务。

当长任务结束时,我需要运行一个管理用户界面的例程。

因此,在异步任务结束时,我执行以下涉及接口的操作,不能直接执行该操作,否则将给出错误。所以我用

this.runOnUiThread(Runnable { x(...)})   // Kotlin

很多时候,这个错误发生在函数x的某个点上。

如果函数x在线程外部被调用

              x(...)  // Kotlin

Android Studio将显示带有错误行的调用堆栈,并且可以在几分钟内轻松解决问题。

由于我的源代码是驯服的,并且不存在严重的结构问题(上面的许多答案都描述了这种错误),所以出现这种可怕的错误消息的原因更加严重,而不那么重要。

这只是链接到线程的执行中的任何愚蠢错误(例如,访问超出定义长度的向量),如下图所示:

           var i = 10                  // Kotlin
           ...
           var arr = Array(5){""}       
           var element = arr[i]       // 10 > 5, it's a index overflow

不幸的是,Android Studio并没有指出这个愚蠢的错误。

我甚至认为这是一个bug,因为Android Studio知道有一个错误,它位于哪里,但是,由于一些未知的原因,它丢失了,并给出了一个随机的消息,完全与问题无关,即,一个奇怪的消息,没有提示显示。

解决方案:有足够的耐心在调试器中一步一步地运行,直到到达错误线,这是Android Studio拒绝提供的。

这种情况在我身上发生过几次,我想这是Android项目中非常常见的错误。在我使用线程之前,我从来没有给过我这种错误。

没有人是绝对正确的,我们容易犯一些小错误。在这种情况下,你不能指望Android Studio的直接帮助来发现你的错误!

我也遇到了同样的问题,并找到了这个页面,虽然我的情况不同,但我在定义警告框之前从if块调用了finish。

所以,简单地调用解散不会工作(因为它还没有做出),但在阅读Alex Volovoy的回答后,意识到这是警报框造成的。我尝试在if块内的结束后添加一个返回语句,这解决了这个问题。

我以为一旦你调用了finish,它就会停止所有的事情,并在那里完成,但它并没有。它似乎走到了它所在的代码块的末尾,然后结束。

如果你想实现一种情况,有时它会在执行一些代码之前完成你必须在完成之后放一个return语句,否则它会继续执行就像在代码块的末尾调用了finish,而不是在你调用它的地方。这就是为什么我得到了那些奇怪的错误。

private picked(File aDirectory){
     if(aDirectory.length()==0){
        setResult(RESULT_CANCELED, new Intent()); 
        finish(); 
        return;
    }
     AlertDialog.Builder alert= new AlertDialog.Builder(this); // Start dialog builder
     alert
        .setTitle("Question")
        .setMessage("Do you want to open that file?"+aDirectory.getName());
    alert
        .setPositiveButton("OK", okButtonListener)
        .setNegativeButton("Cancel", cancelButtonListener);
    alert.show();
}

如果你没有在我调用finish之后立即将return放在那里,它就会像你在alert.show()之后调用它一样;因此,它会说窗口是在你让对话框出现后泄露的,即使不是这样,它仍然认为它是。

我想我应该在这里添加这个,因为这显示了完成命令的作用不同,然后我认为它做到了,我猜在我发现这一点之前,还有其他人和我想的一样。

在我的例子中,原因是我忘记在Android manifest文件中包含权限。

我是怎么知道的?好吧,就像@Bobby在接受的答案下面的评论中所说的那样,只要进一步向上滚动到日志,就会看到真正抛出异常的第一个原因或事件。显然,消息“活动已经泄漏了最初添加的窗口”只是一个异常,无论第一个异常是什么。

当我在AsyncTask中使用ProgressDialog时,这种情况发生在我身上。实际上,我在onPostExecute中使用hide()方法。基于@Alex Volovoy的答案,我需要使用遣散()与ProgressDialog删除它在onPostExecute和它完成。

progressDialog.hide(); // Don't use it, it gives error

progressDialog.dismiss(); // Use it

我在我的视频播放器应用程序中获取了这些日志。这些消息是在视频播放器关闭时抛出的。有趣的是,我曾经以随机的方式在几次运行中获得这些日志。此外,我的应用程序不涉及任何progressdialog。最后,我用下面的实现解决了这个问题。

@Override
protected void onPause()
{
    Log.v("MediaVideo", "onPause");
    super.onPause();
    this.mVideoView.pause();
    this.mVideoView.setVisibility(View.GONE);
}

@Override
protected void onDestroy()
{
    Log.v("MediaVideo", "onDestroy");
    super.onDestroy();
}

@Override
protected void onResume()
{
    Log.v("MediaVideo", "onResume");
    super.onResume();
    this.mVideoView.resume();
}

覆盖OnPause调用mVideoView.pause()和设置可见性为GONE。通过这种方式,我可以解决“活动已泄漏窗口”日志错误问题。