(Android)從表單泄漏談android:configChanges屬性
今天有幸去哥們的大公司做了半天的臨時工,一個偶現的Bug折騰了他好久,好不容易今天抓到了異常Log日誌,大致的意思就是android.view.windowleaked——表單泄漏。我在網上查了資料:
Android的每一個Activity都有個WindowManager表單管理器,構建在某個Activity之上的對話方塊、PopupWindow也有相應的WindowManager表單管理器。因為Dialog、PopupWindown不能脫離Activity而單獨存在著,所以當承載某個Dialog或者某個PopupWindow正在顯示的Activity被finish()後,而Dialog(或PopupWindow)沒有正常退出的話,就會拋Window Leaked錯誤了,因為這個Dialog(或PopupWindow)的WindowManager已經沒有誰可以附屬了,所以它的表單管理器就泄漏了。
根據此資訊分析出,在進入新的Activity時突然轉屏(哥們開發的sdk支援橫豎屏切換),因為在AndroidManifest.xml中沒有配置android:configChanges屬性,此時Activity會重新調用onCreate方法,即會重新調用整個生命週期,而此時的Dialog已經顯示並沒有dismiss,所以造成了表單泄漏。解決的方法就變得如此簡單,在AndroidManifest.xml中配置android:configChanges屬性,這樣當我們橫豎屏切換的時候會調用Activity的onConfigurationChanged方法,不會重新調用整個生命週期了。我們最後配置了android:configChanges="screenSize|orientation|keyboardHidden|navigation"。
既然談到了android:configChanges屬性,我又做了進一步的研究,綜合網上的資料,總結出:
1、不設定Activity的android:configChanges時,切屏會重新調用整個生命週期,切橫屏時會執行一次,切豎屏時會執行兩次
2、設定Activity的android:configChanges="orientation"時,切屏還是會重新調用整個生命週期,切橫、豎屏時只會執行一次
3、設定Activity的android:configChanges="orientation|keyboardHidden"時,切屏不會重新調用整個生命週期,只會執行onConfigurationChanged方法
但是,自從Android 3.2(API 13),在設定Activity的android:configChanges="orientation|keyboardHidden"後,還是一樣會重新調用各個生命週期的。因為screensize也開始跟著裝置的橫豎切換而改變。所以在AndroidManifest.xml裡設定的MiniSdkVersion和 TargetSdkVersion屬性大於等於13的情況下,如果你想阻止程式在運行時重新載入Activity,除了設定"orientation", 你還必須設定" screenSize"。
——以上資訊從網上看到,覺得很有用,但自己並沒有驗證過,不過相信也是LZ驗證過發出的,應該會很有用的。
附上android:configChanges屬性解釋:
| VALUE |
DESCRIPTION |
| "mcc" |
國際移動使用者識別碼所屬國家代號是改變了----- sim被偵測到了,去更新mcc mcc是移動使用者所屬國家代號 |
| "mnc" |
國際移動使用者識別碼的移動網號碼是改變了------ sim被偵測到了,去更新mnc MNC是移動網號碼,最多由兩位元字組成,用於識別移動使用者所歸屬的移動通訊網 |
| "locale" |
地址改變了-----使用者選擇了一個新的語言會顯示出來 |
| "touchscreen" |
觸控螢幕是改變了------通常是不會發生的 |
| "keyboard" |
鍵盤發生了改變----例如使用者用了外部的鍵盤 |
| "keyboardHidden" |
鍵盤的可用性發生了改變 |
| "navigation" |
導航發生了變化-----通常也不會發生 |
| "screenLayout" |
螢幕的顯示發生了變化------不同的顯示被啟用 |
| "fontScale" |
字型比例發生了變化----選擇了不同的全域字型 |
| "uiMode" |
使用者的模式發生了變化 |
| "orientation" |
螢幕方向改變了 |
| "screenSize" |
螢幕大小改變了 |
| "smallestScreenSize" |
螢幕的物理大小改變了,如:串連到一個外部的螢幕上 |
今天收穫不小,見識了大公司的霸氣,也加深了android:configChanges屬性的瞭解。