旋轉(zhuǎn)屏幕導(dǎo)致Activity重建
當(dāng)屏幕旋轉(zhuǎn)時(shí),這個(gè)Configuration就發(fā)生了改變,因此當(dāng)前顯示的Activity需要被重建,Activity對象會被終止,它的onPause()、onStop()和onDestroy()方法依次觸發(fā),然后一個(gè)新的Activity對象被創(chuàng)建,onCreate()方法被觸發(fā)。假設(shè)屏幕旋轉(zhuǎn)前,用戶正在手機(jī)上填寫一個(gè)注冊表單,如果處理不當(dāng),用戶會發(fā)現(xiàn)旋轉(zhuǎn)后的表單變成空白的了,嚴(yán)重影響使用體驗(yàn)。
本文引用地址:http://www.ex-cimer.com/article/201609/305275.htm要解決這個(gè)問題有三種方法:
方法1:禁止旋轉(zhuǎn)屏幕
毫無疑問,這是最懶的辦法了,相當(dāng)于回避了本文提出的問題,方法如下看看就好:
android:screenOrientation=portrait
android:label=@string/app_name>
方法2:旋轉(zhuǎn)后恢復(fù)現(xiàn)場
既然Activity會被銷毀,那么我們就可以使用前文介紹過的“持久化/恢復(fù)現(xiàn)場”方法來解決。即在onPause()里將用戶當(dāng)前已經(jīng)輸入的內(nèi)容保存到數(shù)據(jù)庫或Preference,在onCreate()方法里讀取并填充到表單中,這也是官方推薦的方法。
需要補(bǔ)充一點(diǎn),如果Activity重建需要耗費(fèi)大量資源或需要訪問網(wǎng)絡(luò)導(dǎo)致時(shí)間很長,可以實(shí)現(xiàn)onRetainNonConfigurationInstance()方法將所需數(shù)據(jù)先保存到一個(gè)對象里,像下面這樣:
@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的改變會導(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對象不會被銷毀——作為替代,Activity的onConfigurationChanged()方法被觸發(fā),在這里開發(fā)者可以獲取到當(dāng)前的屏幕方向以便做必要的更新。既然這種情況下的Activity不會被銷毀,旋轉(zhuǎn)后Activity里正顯示的信息(例如文本框中的文字)也就不會丟失了。
假如你的應(yīng)用里,橫屏和豎屏使用同一個(gè)layout資源文件,onConfigurationChanged()里甚至可以什么都不做。但如果橫屏與豎屏使用不同的layout資源文件,例如橫屏用res/layout-land/main.xml,豎屏用res/layout-port/main.xml,則必須在onConfigurationChanged()里重新調(diào)用setContentView()方法以便新的layout能夠生效,這時(shí)雖然Activity對象沒有銷毀,但界面上的各種控件都被銷毀重建了,你需要寫額外的代碼來恢復(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)場,可以做到一舉多得;雖然Google不推薦設(shè)置android:configChanges屬性的方式,但如果你的Activity橫向縱向共用同一個(gè)layout文件,方法3無疑是最省事的。
評論