前幾日出現(xiàn)這樣一個Bug是一個RuntimeException,詳細信息是這樣子的:
java.lang.IllegalArgumentException: View not attached to window manager
at android.view.WindowManagerImpl.findViewLocked(WindowManagerImpl.java:356)
at android.view.WindowManagerImpl.removeView(WindowManagerImpl.java:201)
at android.view.Window$LocalWindowManager.removeView(Window.java:400)
at android.app.Dialog.dismissDialog(Dialog.java:268)
at android.app.Dialog.access$000(Dialog.java:69)
at android.app.Dialog$1.run(Dialog.java:103)
at android.app.Dialog.dismiss(Dialog.java:252)
at xxx.onPostExecute(xxx$1.java:xxx)
首
先是Google了下,發(fā)現(xiàn)引發(fā)這個的原因基本上都一致都是Dismiss對話框的時候,Activity已經(jīng)不再存在。常發(fā)生這類Exception的
情形都是,有一個費時的線程操作,需要在顯示一個ProgressDialog,在任務開始的時候顯示一個對話框,然后當任務完成了再Dismiss對話
框,如果在此期間如果Activity因為某種原因被殺掉且又重新啟動了,那么當Dismiss的時候WindowManager檢查發(fā)現(xiàn)Dialog所
屬的Activity已經(jīng)不存在了,所以會報IllegalArgumentException: View not attached to
window manager.
其實此類Exception的一重要的原因是,ProgressDialog的創(chuàng)建顯
示和取消都允許在非UI線程中進程。在Android當中非UI線程是不允許操作UI相關的事情,比如添加移除View等,但是為會么允許創(chuàng)建顯示和取消
對話框呢?而且還有可能引發(fā)此Exception導致應用Crash。
要想避免此類Exception,就要正確的使用對話框,也要正確的使用線程。
正確的使用對話框不要在非UI線程中使用對話框創(chuàng)建,顯示和取消對話框。
那么對于異步操作顯示對話框怎么辦呢?Activity都有相應的操作對話框的回調(diào)比如
onCreateDialog(),showDialog(),dimissDialog(),removeDialog()等等。這些因為都是
Activity的方法,所以用起來更方便,也不用顯示創(chuàng)建和操控Dialog對象,一切都由框架操控,相對來說比較安全。
另外就是一定要讓對話框對象在Activity的可控制范圍之內(nèi)和生命周期之內(nèi),比如一定要是它的成員變量,并且在讓對話框變量活躍在Activity的onCreate()和onDestroy()之間。
正確的使用線程盡量少用單獨線程,除非是真正的費時操作才用線程,線程也不要直接用Java式的匿名線程,除非是那種單純的操作,操作完成后不需要做其他事情的。
盡可能多用Android提供的類比如AsyncTask等。另外如果線程操作過程中還需要與主線程有交互,那么最好保存一個線程的對象,并且線程內(nèi)部最有一定的控制,這樣可以讓Activity更好的操控線程。
如果說某些操作是特別費時的,且是經(jīng)常性的操作,比如從網(wǎng)絡獲取數(shù)據(jù),或是從后臺讀取文件,或是導入/導出,恢復/備份的事情,最好放到后臺Service中去做,然后在StatusBar中給出相應進度。
如有更好的解決方式,請給予指點,謝謝!