前幾日出現(xiàn)這樣一個(gè)Bug是一個(gè)RuntimeException晤柄,詳細(xì)信息是這樣子的:
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ā)這個(gè)的原因基本上都一致都是Dismiss對話框的時(shí)候钦奋,Activity已經(jīng)不再存在。常發(fā)生這類Exception的情形都是正压,有一個(gè)費(fèi)時(shí)的線程操作猜年,需要在顯示一個(gè)ProgressDialog羽杰,在任務(wù)開始的時(shí)候顯示一個(gè)對話框,然后當(dāng)任務(wù)完成了再Dismiss對話框端蛆,如果在此期間如果Activity因?yàn)槟撤N原因被殺掉且又重新啟動(dòng)了凤粗,那么當(dāng)Dismiss的時(shí)候WindowManager檢查發(fā)現(xiàn)Dialog所屬的Activity已經(jīng)不存在了,所以會(huì)報(bào)IllegalArgumentException: View not attached to window manager.
其實(shí)此類Exception的一重要的原因是今豆,ProgressDialog的創(chuàng)建顯示和取消都允許在非UI線程中進(jìn)程侈沪。在Android當(dāng)中非UI線程是不允許操作UI相關(guān)的事情,比如添加移除View等晚凿,但是為會(huì)么允許創(chuàng)建顯示和取消對話框呢亭罪?而且還有可能引發(fā)此Exception導(dǎo)致應(yīng)用Crash。
要想避免此類Exception歼秽,就要正確的使用對話框应役,也要正確的使用線程。
正確的使用對話框
不要在非UI線程中使用對話框創(chuàng)建燥筷,顯示和取消對話框箩祥。
那么對于異步操作顯示對話框怎么辦呢?Activity都有相應(yīng)的操作對話框的回調(diào)比如onCreateDialog(),showDialog(),dimissDialog(),removeDialog()等等肆氓。這些因?yàn)槎际茿ctivity的方法袍祖,所以用起來更方便,也不用顯示創(chuàng)建和操控Dialog對象谢揪,一切都由框架操控蕉陋,相對來說比較安全捐凭。
另外就是一定要讓對話框?qū)ο笤贏ctivity的可控制范圍之內(nèi)和生命周期之內(nèi),比如一定要是它的成員變量凳鬓,并且在讓對話框變量活躍在Activity的onCreate()和onDestroy()之間茁肠。
正確的使用線程 www.2cto.com
盡量少用單獨(dú)線程,除非是真正的費(fèi)時(shí)操作才用線程缩举,線程也不要直接用Java式的匿名線程垦梆,除非是那種單純的操作,操作完成后不需要做其他事情的仅孩。
盡可能多用Android提供的類比如AsyncTask等托猩。另外如果線程操作過程中還需要與主線程有交互,那么最好保存一個(gè)線程的對象辽慕,并且線程內(nèi)部最有一定的控制京腥,這樣可以讓Activity更好的操控線程。
如果說某些操作是特別費(fèi)時(shí)的鼻百,且是經(jīng)常性的操作绞旅,比如從網(wǎng)絡(luò)獲取數(shù)據(jù),或是從后臺(tái)讀取文件温艇,或是導(dǎo)入/導(dǎo)出因悲,恢復(fù)/備份的事情,最好放到后臺(tái)Service中去做勺爱,然后在StatusBar中給出相應(yīng)進(jìn)度晃琳。
---------------------!