一個Bug
前幾日出現這樣一個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了下,發現引發這個的原因基本上都一致都是Dismiss對話方塊的時候,Activity已經不再存在。常發生這類Exception的情形都是,有一個費時的線程操作,需要在顯示一個ProgressDialog,在任務開始的時候顯示一個對話方塊,然後當任務完成了再Dismiss對話方塊,如果在此期間如果Activity因為某種原因被殺掉且又重新啟動了,那麼當Dismiss的時候WindowManager檢查發現Dialog所屬的Activity已經不存在了,所以會報IllegalArgumentException: View not attached to window manager.
其實此類Exception的一重要的原因是,ProgressDialog的建立顯示和取消都允許在非UI線程中進程。在Android當中非UI線程是不允許操作UI相關的事情,比如添加移除View等,但是為會麼允許建立顯示和取消對話方塊呢?而且還有可能引發此Exception導致應用Crash。
要想避免此類Exception,就要正確的使用對話方塊,也要正確的使用線程。
正確的使用對話方塊
不要在非UI線程中使用對話方塊建立,顯示和取消對話方塊。
那麼對於非同步作業顯示對話方塊怎麼辦呢?Activity都有相應的操作對話方塊的回調比如onCreateDialog(),showDialog(),dimissDialog(),removeDialog()等等。這些因為都是Activity的方法,所以用起來更方便,也不用顯示建立和操控Dialog對象,一切都由架構操控,相對來說比較安全。
另外就是一定要讓對話方塊對象在Activity的可控制範圍之內和生命週期之內,比如一定要是它的成員變數,並且在讓對話方塊變數活躍在Activity的onCreate()和onDestroy()之間。
正確的使用線程 www.2cto.com
盡量少用單獨線程,除非是真正的費時操作才用線程,線程也不要直接用Java式的匿名線程,除非是那種單純的操作,操作完成後不需要做其他事情的。
儘可能多用Android提供的類比如AsyncTask等。另外如果線程操作過程中還需要與主線程有互動,那麼最好儲存一個線程的對象,並且線程內部最有一定的控制,這樣可以讓Activity更好的操控線程。
如果說某些操作是特別費時的,且是經常性的操作,比如從網路擷取資料,或是從後台讀取檔案,或是匯入/匯出,恢複/備份的事情,最好放到後台Service中去做,然後在StatusBar中給出相應進度。