旋轉屏幕致使Activity重建

當屏幕旋轉時,這個Configuration就發生了改變,所以當前顯示的Activity須要被重建,Activity對象會被終止,它的onPause()、onStop()和onDestroy()方法依次觸發,而後一個新的Activity對象被建立,onCreate()方法被觸發。假設屏幕旋轉前,用戶正在手機上填寫一個註冊表單,若是處理不當,用戶會發現旋轉後的表單變成空白的了,嚴重影響使用體驗。html

要解決這個問題有三種方法:android

方法1:禁止旋轉屏幕

毫無疑問,這是最懶的辦法了,至關於迴避了本文提出的問題,方法以下看看就好:數據庫

<activity android:name=".MyActivity"
          android:screenOrientation="portrait"
          android:label="@string/app_name">

 

方法2:旋轉後恢復現場

既然Activity會被銷燬,那麼咱們就可使用前文介紹過的「持久化/恢復現場」方法來解決。即在onPause()裏將用戶當前已經輸入的內容保存到數據庫或Preference,在onCreate()方法裏讀取並填充到表單中,這也是官方推薦的方法。網絡

須要補充一點,若是Activity重建須要耗費大量資源或須要訪問網絡致使時間很長,能夠實現onRetainNonConfigurationInstance()方法將所需數據先保存到一個對象裏,像下面這樣:app

@Overridepublic Object onRetainNonConfigurationInstance() {    final MyDataObject data = collectMyLoadedData();    return data;
}

重建時,在onCreate()方法裏經過getLastNonConfigurationInstance()方法得到以前保存的數據,以下所示:ide

@Overridepublic void onCreate(Bundle savedInstanceState) {    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);    final MyDataObject data = (MyDataObject) getLastNonConfigurationInstance();    if (data == null) {//表示不是因爲Configuration改變觸發的onCreate()
        data = loadMyData();
    }
    ...
}

 

方法3:手工處理旋轉

通常狀況下Configuration的改變會致使Activity被銷燬重建,但也有辦法讓指定的Configuration改變時不重建Activity,方法是在AndroidManifest.xml裏經過android:configChanges屬性指定須要忽略的Configuration名字,例以下面這樣:ui

<activity android:name=".MyActivity"
          android:configChanges="orientation|keyboardHidden"
          android:label="@string/app_name">

這樣設置之後,當屏幕旋轉時Activity對象不會被銷燬——做爲替代,Activity的onConfigurationChanged()方法被觸發,在這裏開發者能夠獲取到當前的屏幕方向以便作必要的更新。既然這種狀況下的Activity不會被銷燬,旋轉後Activity里正顯示的信息(例如文本框中的文字)也就不會丟失了。this

假如你的應用裏,橫屏和豎屏使用同一個layout資源文件,onConfigurationChanged()裏甚至能夠什麼都不作。但若是橫屏與豎屏使用不一樣的layout資源文件,例如橫屏用res/layout-land/main.xml,豎屏用res/layout-port/main.xml,則必須在onConfigurationChanged()裏從新調用setContentView()方法以便新的layout可以生效,這時雖然Activity對象沒有銷燬,但界面上的各類控件都被銷燬重建了,你須要寫額外的代碼來恢復界面信息。spa

@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開發文檔不建議使用這種方式處理Configuration改變:rest

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.

 

最佳實踐

考慮到旋轉屏幕並非使Activity被銷燬重建的惟一因素,仍然推薦前文介紹過的方法:在onPause()裏持久化Activity狀態,在onCreate()裏恢復現場,能夠作到一舉多得;雖然Google不推薦設置android:configChanges屬性的方式,但若是你的Activity橫向縱向共用同一個layout文件,方法3無疑是最省事的。

參考資料:

Configuration Changes 
Handling Runtime Changes
Activity restart on rotation Android
How to handle screen orientation change when progress dialog and background thread active?

本文地址:http://www.cnblogs.com/bjzhanghao/archive/2012/11/09/2761897.html

相關文章
相關標籤/搜索