橫豎屏切換導(dǎo)致頁(yè)面頻繁重啟screenLayout解析
前言
前幾天多名用戶反饋同一個(gè)問(wèn)題,在小新平板上無(wú)法上網(wǎng)課,點(diǎn)擊上課按鈕后就退回到首頁(yè)了。同事了解了一下發(fā)現(xiàn)小新平板現(xiàn)在銷量特別好,于是趕緊申請(qǐng)了一臺(tái)測(cè)試機(jī)打算看看到底是什么問(wèn)題。
最后同事發(fā)現(xiàn)是screenLayout的問(wèn)題,在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。
但是為什么在其他設(shè)備上沒(méi)問(wèn)題,唯獨(dú)在小新平板上有問(wèn)題呢?而且為什么添加了screenLayout就解決問(wèn)題了,這其中的原理是什么?我非常好奇,于是自己研究了一下。
android:configChanges
首先我們要知道android:configChanges
這個(gè)配置的作用,這里我們來(lái)看看官方的介紹:
列出 Activity 將自行處理的配置變更。在運(yùn)行時(shí)發(fā)生配置變更時(shí),默認(rèn)情況下會(huì)關(guān)閉 Activity 并將其重啟,但使用該屬性聲明配置將阻止 Activity 重啟。相反,Activity 會(huì)保持運(yùn)行狀態(tài),并且系統(tǒng)會(huì)調(diào)用其 onConfigurationChanged() 方法。
當(dāng)Activity的配置發(fā)生變更時(shí)(如橫豎屏切換),如果在android:configChanges中沒(méi)有添加該配置,那么就會(huì)關(guān)閉并重啟Activity,這時(shí)候debug會(huì)發(fā)現(xiàn)重新執(zhí)行了onCreate。但是當(dāng)我們添加了該配置,如果該配置發(fā)生變更,則不會(huì)重啟Activity,會(huì)調(diào)用onConfigurationChanged()方法。
那么orientation
就是對(duì)應(yīng)著橫豎屏切換,keyboardHidden
則是軟鍵盤彈出,screenSize
則是屏幕尺寸改變。這么來(lái)看我們?cè)O(shè)置了orientation
應(yīng)該就可以了,但是官方在這里有一個(gè)提示,如下:
官方建議在設(shè)置orientation
的同時(shí)設(shè)置screenSize
和screenLayout
,screenSize
我么可以理解,橫豎屏切換時(shí)寬高會(huì)交換,那么screenLayout
是指什么呢?
screenLayout
前面我們知道存在配置時(shí)會(huì)執(zhí)行onConfigurationChanged(),這個(gè)函數(shù)的參數(shù)是Configuration
類型的,這個(gè)類里保存著Activity的配置,我們來(lái)看看對(duì)screenLayout
這個(gè)屬性的描述:
/** 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其實(shí)是承載著四個(gè)配置的:
- 屏幕大小等級(jí):有
SCREENLAYOUT_SIZE_SMALL
,SCREENLAYOUT_SIZE_NORMAL
,SCREENLAYOUT_SIZE_LARGE
和SCREENLAYOUT_SIZE_XLARGE
四種 - 是否寬屏:屏幕是否比普通屏幕更寬或更高
- 屏幕方向:屏幕是從左向右顯示,還是從有向左顯示
- 是否是圓角屏:屏幕是否有圓角
通過(guò)將screenLayout于對(duì)應(yīng)mask進(jìn)行與運(yùn)算就可以得到當(dāng)前屏幕在該屬性的值,比如:
screenLayout & SCREENLAYOUT_SIZE_MASK
就可以得到屏幕大小等級(jí),一定是SCREENLAYOUT_SIZE_SMALL
, SCREENLAYOUT_SIZE_NORMAL
, SCREENLAYOUT_SIZE_LARGE
和SCREENLAYOUT_SIZE_XLARGE
之一。
四個(gè)mask對(duì)應(yīng)的二進(jìn)制分別是:
- SCREENLAYOUT_SIZE_MASK: 0000 0000 1111
- SCREENLAYOUT_LONG_MASK: 0000 0011 0000
- SCREENLAYOUT_LAYOUTDIR_MASK:0000 1100 0000
- SCREENLAYOUT_ROUND_MASK: 0011 0000 0000
比如在我的測(cè)試機(jī)上得到的screenLayout是268435810,轉(zhuǎn)成二進(jìn)制就是
0001 0000 0000 0000 0000 0001 0110 0010
通過(guò)與四個(gè)mask分別計(jì)算得到SCREENLAYOUT_SIZE_NORMAL、SCREENLAYOUT_LONG_NO、SCREENLAYOUT_LAYOUTDIR_LTR和SCREENLAYOUT_ROUND_NO
而且無(wú)論橫屏還是豎屏,我的測(cè)試機(jī)的screenLayout是不變的,所以即使在android:configChanges
中沒(méi)用添加screenLayout,橫豎屏切換的時(shí)候也不會(huì)重啟Activity,因?yàn)槠渌麅蓚€(gè)屬性orientation
和screenSize
都添加了。
小新平板
那么為什么小新平板上會(huì)有不同的現(xiàn)象,在android:configChanges
中添加screenLayout完后我在onConfigurationChanged
函數(shù)中debug獲取screenLayout值,發(fā)現(xiàn)橫屏和豎屏這個(gè)值是不同的,分別是268435812和268435796。
通過(guò)與mask運(yùn)算獲取四個(gè)屬性后對(duì)比發(fā)現(xiàn),橫豎屏切換后SCREENLAYOUT_LONG_NO變成了SCREENLAYOUT_LONG_YES,所以沒(méi)添加screenLayout的時(shí)候就會(huì)導(dǎo)致Activity重啟,添加后就可以了。
至于為什么重啟Activity會(huì)導(dǎo)致回退到首頁(yè),其實(shí)是用戶表述問(wèn)題,現(xiàn)象是應(yīng)用重啟了。為什么應(yīng)用會(huì)重啟,這是因?yàn)槲覀兊膽?yīng)用架構(gòu)是單Activity的,頁(yè)面由fragment承載。當(dāng)從詳情頁(yè)打開(kāi)上課頁(yè)面時(shí),會(huì)通過(guò)代碼手動(dòng)將豎屏切換到橫屏(同時(shí)為了返回詳情頁(yè)時(shí)換回豎屏,在詳情頁(yè)手動(dòng)切換回豎屏)。這時(shí)候Activity重啟并以橫屏狀態(tài)恢復(fù)所有fragment,但是恢復(fù)詳情頁(yè)時(shí)候,又切換到豎屏,所以又重啟;然后以豎屏狀態(tài)恢復(fù)到上課頁(yè)面,又進(jìn)行了切換,于是死循環(huán),最后系統(tǒng)將應(yīng)用重啟。
總結(jié)
一直以來(lái)對(duì)android:configChanges
沒(méi)有深入了解,正好趁著這個(gè)機(jī)會(huì)了解了一番,大家以后一定要注意這里,特別注意官方文檔的提示,很重要。
以上就是橫豎屏切換導(dǎo)致頁(yè)面頻繁重啟screenLayout解析的詳細(xì)內(nèi)容,更多關(guān)于screenLayout頁(yè)面重啟的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android自定義View仿QQ運(yùn)動(dòng)步數(shù)效果
這篇文章主要為大家詳細(xì)介紹了Android自定義View仿QQ運(yùn)動(dòng)步數(shù)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11Android自定義視圖實(shí)現(xiàn)手指移動(dòng)軌跡
這篇文章主要為大家詳細(xì)介紹了Android自定義視圖實(shí)現(xiàn)手指移動(dòng)軌跡,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-06-06Android仿百度地圖小度語(yǔ)音助手的貝塞爾曲線動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Android仿百度地圖小度語(yǔ)音助手的貝塞爾曲線動(dòng)畫(huà),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android自定義View實(shí)現(xiàn)loading動(dòng)畫(huà)加載效果
項(xiàng)目開(kāi)發(fā)中對(duì)Loading的處理是比較常見(jiàn)的,安卓系統(tǒng)提供的不太美觀,引入第三發(fā)又太麻煩,這時(shí)候自己定義View來(lái)實(shí)現(xiàn)這個(gè)效果。這篇文章主要介紹了Android自定義View實(shí)現(xiàn)loading動(dòng)畫(huà)加載效果,需要的朋友可以參考下2017-03-03Android中利用viewflipper動(dòng)畫(huà)切換屏幕效果
這篇文章主要介紹了Android中利用viewflipper動(dòng)畫(huà)切換屏幕效果的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09深入淺出學(xué)習(xí)Android ListView基礎(chǔ)
這篇文章主要介紹了深入淺出的帶領(lǐng)大家學(xué)習(xí)Android ListView基礎(chǔ),ListView是安卓里常用的控件,本文介紹一下常用用法,以及優(yōu)化等方法,感興趣的小伙伴們可以參考一下2016-01-01Android實(shí)現(xiàn)可滑動(dòng)的自定義日歷控件
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)可滑動(dòng)的自定義日歷控件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07