[Android問答] 旋轉螢幕導致Activity重建怎麼辦?

來源:互聯網
上載者:User

Android開發文檔上專門有一小節解釋這個問題。簡單來說,Activity是負責與使用者互動的最主要機制,任何“設定”(Configuration)的改變都可能對Activity的介面造成影響,這時系統會銷毀並重建Activity以便反映新的Configuration。

“螢幕方向”(orientation)是一個Configuration,通過查看Configuration類的javadoc可以看到其他Configuration還有哪些:如fontScale、keyboardHidden和locale等等。

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

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

方法1:禁止旋轉螢幕

毫無疑問,這是最懶的辦法了,相當於迴避了本文提出的問題,方法如下看看就好:

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

 

方法2:旋轉後恢複現場

既然Activity會被銷毀,那麼我們就可以使用前文介紹過的“持久化/恢複現場”方法來解決。即在onPause()裡將使用者當前已經輸入的內容儲存到資料庫或Preference,在onCreate()方法裡讀取並填充到表單中,這也是官方推薦的方法。

需要補充一點,如果Activity重建需要耗費大量資源或需要訪問網路導致時間很長,可以實現onRetainNonConfigurationInstance()方法將所需資料先儲存到一個對象裡,像下面這樣:

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

重建時,在onCreate()方法裡通過getLastNonConfigurationInstance()方法獲得之前儲存的資料,如下所示:

@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名字,例如下面這樣:

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

這樣設定以後,當旋轉螢幕時Activity對象不會被銷毀——作為替代,Activity的onConfigurationChanged()方法被觸發,在這裡開發人員可以擷取到當前的螢幕方向以便做必要的更新。既然這種情況下的Activity不會被銷毀,旋轉後Activity裡正顯示的資訊(例如文字框中的文字)也就不會丟失了。

假如你的應用裡,橫屏和豎屏使用同一個layout資源檔,onConfigurationChanged()裡甚至可以什麼都不做。但如果橫屏與豎屏使用不同的layout資源檔,例如橫屏用res/layout-land/main.xml,豎屏用res/layout-port/main.xml,則必須在onConfigurationChanged()裡重新調用setContentView()方法以便新的layout能夠生效,這時雖然Activity對象沒有銷毀,但介面上的各種控制項都被銷毀重建了,你需要寫額外的代碼來恢複介面資訊。

@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改變:

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?

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.