错误场景:
在我的Activity中我使用了ProgressDialog,用于在执行耗时操作时显示进度条让用户等待。耗时任务(保存图片)放在AsyncTask中异步执行,当操作执行完成时,让进度条消失。
错误:
出现类似下边的错误,提示窗口泄露
05-31 10:46:37.658: ERROR/WindowManager(21946): android.view.WindowLeaked: Activity *** has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@2b0c2100 that was originally added here
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.view.ViewRoot.<init>(ViewRoot.java:264)
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:148)
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:91)
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.view.Window$LocalWindowManager.addView(Window.java:424)
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.app.Dialog.show(Dialog.java:241)
05-31 10:46:37.658: ERROR/WindowManager(21946): at android.app.AlertDialog$Builder.show(AlertDialog.java:816)
从异常描述中,大致的意思是存在窗口句柄泄露,即未能及时销毁某个PhoneWindow。而这往往误导了我们,把过多的精力放在查找所谓的内存泄露上了。其实存在这么一种情况,即因我们在非主线程中的某些操作不当而产生了一个严重的异常,从而强制当前Activity被关闭。而在关闭的同时,却没能及时的调用dismiss来解除对 ProgressDialog等的引用,从而系统抛出了标题中的错误,而掩盖了真正导致这个错误的异常信息。
解决:
我们可以重写Activity的onDestroy方法,在方法中调用dismiss来解除对ProgressDialog等的引用。这样就能看到真正出错的原因。我错误真正原因是在图片处理时发生了OOM,导致Activity关闭,从而窗口泄露。
类似错误:
http://myhpu2008.iteye.com/blog/1066085
由于在Activity中创建并显示了Alertdialog,但并未使用Activity提供的showDialog()方法,因此导致在Activity被结束后,Alertdialog所引用的context为空。解决办法为,1,将dialog采用oncreateDialog的方式创建,交由系统维护。
2.在onDestroy()方法中将该alertdialog dismiss掉。
http://www.dewen.io/q/3334
主要是由于在屏幕旋转的时候,view发生改变,而另一个进程还在尝试改变老的视图,去掉diaolog.
我用两个方式,都比较简单:
1.是在配置文件中忽略横竖屏旋转。