Android開發文檔上專門有一小節解釋這個問題。簡單來說,Activity是負責與使用者互動的最主要機制,任何“設定”(Configuration)的改變都可能對Activity的界面造成影響,這時系統會銷毀并重建Activity以便反映新的Configuration。
“螢幕方向”(orientation)是一個Configuration,通過檢視Configuration類的javadoc可以看到其他Configuration還有哪些:如fontScale、keyboardHidden和locale等等。
當螢幕旋轉時,這個Configuration就發生了改變,是以目前顯示的Activity需要被重建,Activity對象會被終止,它的onPause()、onStop()和onDestroy()方法依次觸發,然後一個新的Activity對象被建立,onCreate()方法被觸發。假設螢幕旋轉前,使用者正在手機上填寫一個系統資料庫單,如果處理不當,使用者會發現旋轉後的表單變成空白的了,嚴重影響使用體驗。
要解決這個問題有三種方法:
方法1:禁止旋轉螢幕
毫無疑問,這是最懶的辦法了,相當于回避了本文提出的問題,方法如下看看就好:
1
2
3
<code><</code><code>activity</code> <code>android:name</code><code>=</code><code>".MyActivity"</code>
<code>android:screenOrientation</code><code>=</code><code>"portrait"</code>
<code>android:label</code><code>=</code><code>"@string/app_name"</code><code>></code>
方法2:旋轉後恢複現場
既然Activity會被銷毀,那麼我們就可以使用前文介紹過的“持久化/恢複現場”方法來解決。即在onPause()裡将使用者目前已經輸入的内容儲存到資料庫或Preference,在onCreate()方法裡讀取并填充到表單中,這也是官方推薦的方法。
需要補充一點,如果Activity重建需要耗費大量資源或需要通路網絡導緻時間很長,可以實作onRetainNonConfigurationInstance()方法将所需資料先儲存到一個對象裡,像下面這樣:
4
5
<code>@Override</code>
<code>public</code> <code>Object onRetainNonConfigurationInstance() {</code>
<code>final</code> <code>MyDataObject data = collectMyLoadedData();</code>
<code>return</code> <code>data;</code>
<code>}</code>
重建時,在onCreate()方法裡通過getLastNonConfigurationInstance()方法獲得之前儲存的資料,如下所示:
6
7
8
9
10
<code>public</code> <code>void</code> <code>onCreate(Bundle savedInstanceState) {</code>
<code>super</code><code>.onCreate(savedInstanceState);</code>
<code>setContentView(R.layout.main);</code>
<code>final</code> <code>MyDataObject data = (MyDataObject) getLastNonConfigurationInstance();</code>
<code>if</code> <code>(data == </code><code>null</code><code>) {</code><code>//表示不是由于Configuration改變觸發的onCreate()</code>
<code>data = loadMyData();</code>
<code>...</code>
方法3:手工處理旋轉
一般情況下Configuration的改變會導緻Activity被銷毀重建,但也有辦法讓指定的Configuration改變時不重建Activity,方法是在AndroidManifest.xml裡通過android:configChanges屬性指定需要忽略的Configuration名字,例如下面這樣:
<code>android:configChanges</code><code>=</code><code>"orientation|keyboardHidden"</code>
這樣設定以後,當螢幕旋轉時Activity對象不會被銷毀——作為替代,Activity的onConfigurationChanged()方法被觸發,在這裡開發者可以擷取到目前的螢幕方向以便做必要的更新。既然這種情況下的Activity不會被銷毀,旋轉後Activity裡正顯示的資訊(例如文本框中的文字)也就不會丢失了。
假如你的應用裡,橫屏和豎屏使用同一個layout資源檔案,onConfigurationChanged()裡甚至可以什麼都不做。但如果橫屏與豎屏使用不同的layout資源檔案,例如橫屏用res/layout-land/main.xml,豎屏用res/layout-port/main.xml,則必須在onConfigurationChanged()裡重新調用setContentView()方法以便新的layout能夠生效,這時雖然Activity對象沒有銷毀,但界面上的各種控件都被銷毀重建了,你需要寫額外的代碼來恢複界面資訊。
<code>public</code> <code>void</code> <code>onConfigurationChanged(Configuration newConfig) {</code>
<code>super</code><code>.onConfigurationChanged(newConfig);</code>
<code>// Checks the orientation of the screen</code>
<code>if</code> <code>(newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) {</code>
<code>Toast.makeText(</code><code>this</code><code>, </code><code>"橫屏模式"</code><code>, Toast.LENGTH_SHORT).show();</code>
<code>} </code><code>else</code> <code>if</code> <code>(newConfig.orientation == Configuration.ORIENTATION_PORTRAIT){</code>
<code>Toast.makeText(</code><code>this</code><code>, </code><code>"豎屏模式"</code><code>, Toast.LENGTH_SHORT).show();</code>
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無疑是最省事的。
本文轉自 我不會抽煙 51CTO部落格,原文連結:http://blog.51cto.com/zhouhongyu1989/1284497,如需轉載請自行聯系原作者