橫豎屏切換導致頁面頻繁重啟screenLayout解析
前言
前幾天多名用戶反饋同一個問題,在小新平板上無法上網(wǎng)課,點擊上課按鈕后就退回到首頁了。同事了解了一下發(fā)現(xiàn)小新平板現(xiàn)在銷量特別好,于是趕緊申請了一臺測試機打算看看到底是什么問題。
最后同事發(fā)現(xiàn)是screenLayout的問題,在manifest中為需要橫豎屏切換的Acitivty配置screenLayout即可,如下:
<activity android:name=".MainActivity" android:configChanges="orientation|keyboardHidden|screenSize|screenLayout" android:launchMode="singleTask"> <intent-filter> ... </intent-filter> </activity>
我們之前android:configChanges配置是orientation|keyboardHidden|screenSize
,缺少了screenLayout。
但是為什么在其他設備上沒問題,唯獨在小新平板上有問題呢?而且為什么添加了screenLayout就解決問題了,這其中的原理是什么?我非常好奇,于是自己研究了一下。
android:configChanges
首先我們要知道android:configChanges
這個配置的作用,這里我們來看看官方的介紹:
列出 Activity 將自行處理的配置變更。在運行時發(fā)生配置變更時,默認情況下會關閉 Activity 并將其重啟,但使用該屬性聲明配置將阻止 Activity 重啟。相反,Activity 會保持運行狀態(tài),并且系統(tǒng)會調用其 onConfigurationChanged() 方法。
當Activity的配置發(fā)生變更時(如橫豎屏切換),如果在android:configChanges中沒有添加該配置,那么就會關閉并重啟Activity,這時候debug會發(fā)現(xiàn)重新執(zhí)行了onCreate。但是當我們添加了該配置,如果該配置發(fā)生變更,則不會重啟Activity,會調用onConfigurationChanged()方法。
那么orientation
就是對應著橫豎屏切換,keyboardHidden
則是軟鍵盤彈出,screenSize
則是屏幕尺寸改變。這么來看我們設置了orientation
應該就可以了,但是官方在這里有一個提示,如下:
官方建議在設置orientation
的同時設置screenSize
和screenLayout
,screenSize
我么可以理解,橫豎屏切換時寬高會交換,那么screenLayout
是指什么呢?
screenLayout
前面我們知道存在配置時會執(zhí)行onConfigurationChanged(),這個函數(shù)的參數(shù)是Configuration
類型的,這個類里保存著Activity的配置,我們來看看對screenLayout
這個屬性的描述:
/** Bit mask of overall layout of the screen. Currently there are four fields: The SCREENLAYOUT_SIZE_MASK bits define the overall size of the screen. They may be one of SCREENLAYOUT_SIZE_SMALL, SCREENLAYOUT_SIZE_NORMAL, SCREENLAYOUT_SIZE_LARGE, or SCREENLAYOUT_SIZE_XLARGE. The SCREENLAYOUT_LONG_MASK defines whether the screen is wider/taller than normal. They may be one of SCREENLAYOUT_LONG_NO or SCREENLAYOUT_LONG_YES. The SCREENLAYOUT_LAYOUTDIR_MASK defines whether the screen layout is either LTR or RTL. They may be one of SCREENLAYOUT_LAYOUTDIR_LTR or SCREENLAYOUT_LAYOUTDIR_RTL. The SCREENLAYOUT_ROUND_MASK defines whether the screen has a rounded shape. They may be one of SCREENLAYOUT_ROUND_NO or SCREENLAYOUT_ROUND_YES. See Supporting Multiple Screens for more information. **/ public int screenLayout;
可以看到screenLayout其實是承載著四個配置的:
- 屏幕大小等級:有
SCREENLAYOUT_SIZE_SMALL
,SCREENLAYOUT_SIZE_NORMAL
,SCREENLAYOUT_SIZE_LARGE
和SCREENLAYOUT_SIZE_XLARGE
四種 - 是否寬屏:屏幕是否比普通屏幕更寬或更高
- 屏幕方向:屏幕是從左向右顯示,還是從有向左顯示
- 是否是圓角屏:屏幕是否有圓角
通過將screenLayout于對應mask進行與運算就可以得到當前屏幕在該屬性的值,比如:
screenLayout & SCREENLAYOUT_SIZE_MASK
就可以得到屏幕大小等級,一定是SCREENLAYOUT_SIZE_SMALL
, SCREENLAYOUT_SIZE_NORMAL
, SCREENLAYOUT_SIZE_LARGE
和SCREENLAYOUT_SIZE_XLARGE
之一。
四個mask對應的二進制分別是:
- SCREENLAYOUT_SIZE_MASK: 0000 0000 1111
- SCREENLAYOUT_LONG_MASK: 0000 0011 0000
- SCREENLAYOUT_LAYOUTDIR_MASK:0000 1100 0000
- SCREENLAYOUT_ROUND_MASK: 0011 0000 0000
比如在我的測試機上得到的screenLayout是268435810,轉成二進制就是
0001 0000 0000 0000 0000 0001 0110 0010
通過與四個mask分別計算得到SCREENLAYOUT_SIZE_NORMAL、SCREENLAYOUT_LONG_NO、SCREENLAYOUT_LAYOUTDIR_LTR和SCREENLAYOUT_ROUND_NO
而且無論橫屏還是豎屏,我的測試機的screenLayout是不變的,所以即使在android:configChanges
中沒用添加screenLayout,橫豎屏切換的時候也不會重啟Activity,因為其他兩個屬性orientation
和screenSize
都添加了。
小新平板
那么為什么小新平板上會有不同的現(xiàn)象,在android:configChanges
中添加screenLayout完后我在onConfigurationChanged
函數(shù)中debug獲取screenLayout值,發(fā)現(xiàn)橫屏和豎屏這個值是不同的,分別是268435812和268435796。
通過與mask運算獲取四個屬性后對比發(fā)現(xiàn),橫豎屏切換后SCREENLAYOUT_LONG_NO變成了SCREENLAYOUT_LONG_YES,所以沒添加screenLayout的時候就會導致Activity重啟,添加后就可以了。
至于為什么重啟Activity會導致回退到首頁,其實是用戶表述問題,現(xiàn)象是應用重啟了。為什么應用會重啟,這是因為我們的應用架構是單Activity的,頁面由fragment承載。當從詳情頁打開上課頁面時,會通過代碼手動將豎屏切換到橫屏(同時為了返回詳情頁時換回豎屏,在詳情頁手動切換回豎屏)。這時候Activity重啟并以橫屏狀態(tài)恢復所有fragment,但是恢復詳情頁時候,又切換到豎屏,所以又重啟;然后以豎屏狀態(tài)恢復到上課頁面,又進行了切換,于是死循環(huán),最后系統(tǒng)將應用重啟。
總結
一直以來對android:configChanges
沒有深入了解,正好趁著這個機會了解了一番,大家以后一定要注意這里,特別注意官方文檔的提示,很重要。
以上就是橫豎屏切換導致頁面頻繁重啟screenLayout解析的詳細內容,更多關于screenLayout頁面重啟的資料請關注腳本之家其它相關文章!
相關文章
Android自定義View實現(xiàn)loading動畫加載效果
項目開發(fā)中對Loading的處理是比較常見的,安卓系統(tǒng)提供的不太美觀,引入第三發(fā)又太麻煩,這時候自己定義View來實現(xiàn)這個效果。這篇文章主要介紹了Android自定義View實現(xiàn)loading動畫加載效果,需要的朋友可以參考下2017-03-03