旋轉(zhuǎn)屏幕導(dǎo)致Activity重建
掃描二維碼
隨時(shí)隨地手機(jī)看文章
當(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ī)上填寫一個(gè)注冊(cè)表單,如果處理不當(dāng),用戶會(huì)發(fā)現(xiàn)旋轉(zhuǎn)后的表單變成空白的了,嚴(yán)重影響使用體驗(yàn)。
要解決這個(gè)問題有三種方法:
方法1:禁止旋轉(zhuǎn)屏幕
毫無疑問,這是最懶的辦法了,相當(dāng)于回避了本文提出的問題,方法如下看看就好:
android:screenOrientation="portrait"
android:label="@string/app_name">
方法2:旋轉(zhuǎn)后恢復(fù)現(xiàn)場(chǎng)
既然Activity會(huì)被銷毀,那么我們就可以使用前文介紹過的“持久化/恢復(fù)現(xiàn)場(chǎng)”方法來解決。即在onPause()里將用戶當(dāng)前已經(jīng)輸入的內(nèi)容保存到數(shù)據(jù)庫或Preference,在onCreate()方法里讀取并填充到表單中,這也是官方推薦的方法。
需要補(bǔ)充一點(diǎn),如果Activity重建需要耗費(fèi)大量資源或需要訪問網(wǎng)絡(luò)導(dǎo)致時(shí)間很長(zhǎng),可以實(shí)現(xiàn)onRetainNonConfigurationInstance()方法將所需數(shù)據(jù)先保存到一個(gè)對(duì)象里,像下面這樣:
@Overridepublic Object onRetainNonConfigurationInstance() { final MyDataObject data = collectMyLoadedData(); return data;
}
重建時(shí),在onCreate()方法里通過getLastNonConfigurationInstance()方法獲得之前保存的數(shù)據(jù),如下所示:
@Overridepublic 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里通過android:configChanges屬性指定需要忽略的Configuration名字,例如下面這樣:
android:configChanges="orientation|keyboardHidden"
android:label="@string/app_name">
這樣設(shè)置以后,當(dāng)屏幕旋轉(zhuǎn)時(shí)Activity對(duì)象不會(huì)被銷毀——作為替代,Activity的onConfigurationChanged()方法被觸發(fā),在這里開發(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ì)象沒有銷毀,但界面上的各種控件都被銷毀重建了,你需要寫額外的代碼來恢復(fù)界面信息。
@Overridepublic 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開發(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被銷毀重建的唯一因素,仍然推薦前文介紹過的方法:在onPause()里持久化Activity狀態(tài),在onCreate()里恢復(fù)現(xiàn)場(chǎng),可以做到一舉多得;雖然Google不推薦設(shè)置android:configChanges屬性的方式,但如果你的Activity橫向縱向共用同一個(gè)layout文件,方法3無疑是最省事的。