Android開(kāi)發(fā) 旋轉(zhuǎn)屏幕導(dǎo)致Activity重建解決方法
“屏幕方向”(orientation)是一個(gè)Configuration,通過(guò)查看Configuration類的javadoc可以看到其他Configuration還有哪些:如fontScale、keyboardHidden和locale等等。
當(dāng)屏幕旋轉(zhuǎn)時(shí),這個(gè)Configuration就發(fā)生了改變,因此當(dāng)前顯示的Activity需要被重建,Activity對(duì)象會(huì)被終止,它的onPause()、onStop()和onDestroy()方法依次觸發(fā),然后一個(gè)新的Activity對(duì)象被創(chuàng)建,onCreate()方法被觸發(fā)。假設(shè)屏幕旋轉(zhuǎn)前,用戶正在手機(jī)上填寫(xiě)一個(gè)注冊(cè)表單,如果處理不當(dāng),用戶會(huì)發(fā)現(xiàn)旋轉(zhuǎn)后的表單變成空白的了,嚴(yán)重影響使用體驗(yàn)。
要解決這個(gè)問(wèn)題有三種方法:
方法1:禁止旋轉(zhuǎn)屏幕
毫無(wú)疑問(wèn),這是最懶的辦法了,相當(dāng)于回避了本文提出的問(wèn)題,方法如下看看就好:
<activity android:name=".MyActivity"
android:screenOrientation="portrait"
android:label="@string/app_name">
方法2:旋轉(zhuǎn)后恢復(fù)現(xiàn)場(chǎng)
既然Activity會(huì)被銷毀,那么我們就可以使用前文介紹過(guò)的“持久化/恢復(fù)現(xiàn)場(chǎng)”方法來(lái)解決。即在onPause()里將用戶當(dāng)前已經(jīng)輸入的內(nèi)容保存到數(shù)據(jù)庫(kù)或Preference,在onCreate()方法里讀取并填充到表單中,這也是官方推薦的方法。
需要補(bǔ)充一點(diǎn),如果Activity重建需要耗費(fèi)大量資源或需要訪問(wèn)網(wǎng)絡(luò)導(dǎo)致時(shí)間很長(zhǎng),可以實(shí)現(xiàn)onRetainNonConfigurationInstance()方法將所需數(shù)據(jù)先保存到一個(gè)對(duì)象里,像下面這樣:
@Override
public Object onRetainNonConfigurationInstance() {
final MyDataObject data = collectMyLoadedData();
return data;
}
重建時(shí),在onCreate()方法里通過(guò)getLastNonConfigurationInstance()方法獲得之前保存的數(shù)據(jù),如下所示:
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
final MyDataObject data = (MyDataObject) getLastNonConfigurationInstance();
if (data == null) {//表示不是由于Configuration改變觸發(fā)的onCreate()
data = loadMyData();
}
...
}
方法3:手工處理旋轉(zhuǎn)
一般情況下Configuration的改變會(huì)導(dǎo)致Activity被銷毀重建,但也有辦法讓指定的Configuration改變時(shí)不重建Activity,方法是在AndroidManifest.xml里通過(guò)android:configChanges屬性指定需要忽略的Configuration名字,例如下面這樣:
<activity android:name=".MyActivity"
android:configChanges="orientation|keyboardHidden"
android:label="@string/app_name">
這樣設(shè)置以后,當(dāng)屏幕旋轉(zhuǎn)時(shí)Activity對(duì)象不會(huì)被銷毀——作為替代,Activity的onConfigurationChanged()方法被觸發(fā),在這里開(kāi)發(fā)者可以獲取到當(dāng)前的屏幕方向以便做必要的更新。既然這種情況下的Activity不會(huì)被銷毀,旋轉(zhuǎn)后Activity里正顯示的信息(例如文本框中的文字)也就不會(huì)丟失了。
假如你的應(yīng)用里,橫屏和豎屏使用同一個(gè)layout資源文件,onConfigurationChanged()里甚至可以什么都不做。但如果橫屏與豎屏使用不同的layout資源文件,例如橫屏用res/layout-land/main.xml,豎屏用res/layout-port/main.xml,則必須在onConfigurationChanged()里重新調(diào)用setContentView()方法以便新的layout能夠生效,這時(shí)雖然Activity對(duì)象沒(méi)有銷毀,但界面上的各種控件都被銷毀重建了,你需要寫(xiě)額外的代碼來(lái)恢復(fù)界面信息。
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
// Checks the orientation of the screen
if (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) {
Toast.makeText(this, "橫屏模式", Toast.LENGTH_SHORT).show();
} else if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT){
Toast.makeText(this, "豎屏模式", Toast.LENGTH_SHORT).show();
}
}
官方的Android開(kāi)發(fā)文檔不建議使用這種方式處理Configuration改變:
Note: Using this attribute should be avoided and used only as a last-resort. Please read Handling Runtime Changes for more information about how to properly handle a restart due to a configuration change.
最佳實(shí)踐
考慮到旋轉(zhuǎn)屏幕并不是使Activity被銷毀重建的唯一因素,仍然推薦前文介紹過(guò)的方法:在onPause()里持久化Activity狀態(tài),在onCreate()里恢復(fù)現(xiàn)場(chǎng),可以做到一舉多得;雖然Google不推薦設(shè)置android:configChanges屬性的方式,但如果你的Activity橫向縱向共用同一個(gè)layout文件,方法3無(wú)疑是最省事的。
- Android實(shí)現(xiàn)屏幕旋轉(zhuǎn)四個(gè)方向準(zhǔn)確監(jiān)聽(tīng)
- Android如何監(jiān)聽(tīng)屏幕旋轉(zhuǎn)
- Android屏幕旋轉(zhuǎn)之橫屏豎屏切換的實(shí)現(xiàn)
- Android6.0開(kāi)發(fā)中屏幕旋轉(zhuǎn)原理與流程分析
- Android webview旋轉(zhuǎn)屏幕導(dǎo)致頁(yè)面重新加載問(wèn)題解決辦法
- Android屏幕旋轉(zhuǎn) 處理Activity與AsyncTask的最佳解決方案
- 詳解Android中Runtime解決屏幕旋轉(zhuǎn)問(wèn)題(推薦)
- Android實(shí)現(xiàn)屏幕旋轉(zhuǎn)方法總結(jié)
- Android實(shí)現(xiàn)簡(jiǎn)單旋轉(zhuǎn)動(dòng)畫(huà)
- Android App獲取屏幕旋轉(zhuǎn)角度的方法
相關(guān)文章
Android編程實(shí)現(xiàn)兩個(gè)Activity之間共享數(shù)據(jù)及互相訪問(wèn)的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)兩個(gè)Activity之間共享數(shù)據(jù)及互相訪問(wèn)的方法,簡(jiǎn)單分析了Android中Activity數(shù)據(jù)共享與訪問(wèn)的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11Flutter實(shí)現(xiàn)倒計(jì)時(shí)秒數(shù)轉(zhuǎn)時(shí)分秒然后倒計(jì)時(shí)功能
有一個(gè)需求,需要在頁(yè)面進(jìn)行顯示倒計(jì)時(shí),倒計(jì)時(shí)結(jié)束后,做相應(yīng)的邏輯處理,這篇文章主要介紹了Flutter實(shí)現(xiàn)倒計(jì)時(shí)功能,秒數(shù)轉(zhuǎn)時(shí)分秒,然后倒計(jì)時(shí),需要的朋友可以參考下2023-08-08關(guān)于Android?Webview?設(shè)置Cookie問(wèn)題詳解
大家好,本篇文章是關(guān)于Android?Webview?設(shè)置Cookie問(wèn)題詳解,感興趣的同學(xué)可以看看,希望對(duì)你起到幫助,有用的話記得收藏,方便下次瀏覽2021-11-11Android EditText輸入框?qū)崿F(xiàn)下拉且保存最近5個(gè)歷史記錄思路詳解
今天給大家介紹Android EditText輸入框?qū)崿F(xiàn)下拉且保存最近5個(gè)歷史記錄功能,android實(shí)現(xiàn)文本框下拉利用sharedpreferences來(lái)保存每次app啟動(dòng)和關(guān)閉時(shí)已經(jīng)填寫(xiě)的數(shù)值,具體代碼跟隨小編一起看看吧2021-07-07Android實(shí)現(xiàn)仿微信tab高亮icon粘著手的滑動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)仿微信tab高亮icon粘著手的滑動(dòng)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08怎樣刪除android的gallery中的圖片實(shí)例說(shuō)明
長(zhǎng)按gallery中的圖片進(jìn)行刪除該圖片的操作,具體實(shí)現(xiàn)如下,感興趣的朋友可以參考下哈2013-06-06Android開(kāi)發(fā)之sqlite3命令行簡(jiǎn)單使用方法
這篇文章主要介紹了Android開(kāi)發(fā)之sqlite3命令行簡(jiǎn)單使用方法,分析了Android增刪改查等常用sqlite3的數(shù)據(jù)庫(kù)操作命令使用方法,需要的朋友可以參考下2016-02-02詳解如何實(shí)現(xiàn)一個(gè)Kotlin函數(shù)類型
這篇文章主要為大家詳細(xì)介紹了如何實(shí)現(xiàn)一個(gè)Kotlin函數(shù)類型,文中的實(shí)現(xiàn)方法講解詳細(xì),具有一定的借鑒價(jià)值,需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-10-10