Activity橫豎屏切換的那些事(生命周期,數據恢復,防止重建)

hocyber 8年前發布 | 26K 次閱讀 Activity Android開發 移動開發

Activity的生命周期,這是每個Android開發者必須了解的知識。Activity是四大組件之一,而且是使用最頻繁的組件。橫豎屏切換是每個Android開發者都會遇到的問題。那么橫豎屏切換后Activity到底發生了什么呢?

1、生命周期的變化

建一個Activity,重寫所有的生命周期方法,然后在這些方法中添加Log。

public class ActivityA extends Activity {

   @Override
   protected void onCreate(Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
     setContentView(R.layout.test_layout);
     Log.i("axe.mg","onCreate()");
   }

   @Override
   protected void onRestart() {
     super.onRestart();
     Log.i("axe.mg","onRestart()");
   }

   @Override
   protected void onStart() {
     super.onStart();
     Log.i("axe.mg","onStart()");
   }

   @Override
   protected void onResume() {
     super.onResume();
     Log.i("axe.mg","onResume()");
   }

   @Override
   protected void onPause() {
     super.onPause();
     Log.i("axe.mg","onPause()");
   }

   @Override
   protected void onStop() {
     super.onStop();
     Log.i("axe.mg","onStop()");
   }

   @Override
   protected void onDestroy() {
     super.onDestroy();
     Log.i("axe.mg","onDestroy()");
   }
}

正常啟動這個Activity。

09-28 23:16:49.809 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onCreate()
09-28 23:16:49.809 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onStart()
09-28 23:16:49.819 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onResume()

通過Log可以看到Activity從創建到展示的的生命周期: onCreate() --->onStart() --->onResume()

進行橫豎屏切換。

09-28 23:17:42.519 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onPause()
09-28 23:17:42.519 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onStop()
09-28 23:17:42.519 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onDestroy()

09-28 23:17:42.719 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onCreate()
09-28 23:17:42.729 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onStart()
09-28 23:17:42.729 24348-24348/com.mg.axe.androiddevelop I/axe.mg: onResume()

通過Log將生命周期分為兩部分來分析:

1、Activity銷毀: onPause() ---> onStop() ---> onDestroy()

2、Activity的重新創建展示 :onCreate() ---> onStart() --->onResume()

結論:當Activity橫豎屏切換時.Activity就會重新創建。
橫豎屏切換時Activity會被銷毀 , onPause() onStop() onDestory() 會被調用。
然后Activity又會被重新創建,onCreate() onStart() onResume() 會被調用。

2、橫豎屏切換后的數據恢復

已經知道Activity橫豎屏切換時Activity會重新創建。那么如何恢復重建前的數據呢?

Activity中有兩個方法用來保存和恢復Activity重建前的數據:

@Override
  protected void onSaveInstanceState(Bundle outState) {
     super.onSaveInstanceState(outState);
  }

  @Override
  protected void onRestoreInstanceState(Bundle savedInstanceState) {
     super.onRestoreInstanceState(savedInstanceState);
  }

onSaveInstanceState:橫豎屏切換時,Activity銷毀前系統會調用onSaveInstanceState通過保存Bundle參數來保存當前的Activity的數據。這個方法會在onStop前調用。

onRestoreInstanceState:當Activity被重新創建之后會調用onRestoreInstanceState,并把Activity銷毀時 onSaveInstanceState方法所保留的數據作為Bundle參數同時傳遞給onRestoreInstanceState和onCreate方法。所以我們可以在onRestoreInstanceState和onCreate方法中看到兩個一樣的參數Bundle savedInstanceState。這個方法會在onStart后調用。

強調一點:必須是Activity異常情況下被終止(例如:橫豎屏切換)才會調用這兩個方法。

通過代碼驗證:在Activity中重寫onSaveInstanceState和onRestoreInstanceState來保存和恢復Activity重建前的數據。

在onSaveInstanceState保存一個“value_string”的string

在onRestoreInstanceState獲取“value_string”這個值。

public class ActivityA extends Activity {

   @Override
   protected void onSaveInstanceState(Bundle outState) {
     super.onSaveInstanceState(outState);
     outState.putString("key","value_string");
     Log.i("axe.mg","onSaveInstanceState");
   }

   @Override
   protected void onRestoreInstanceState(Bundle savedInstanceState) {
     super.onRestoreInstanceState(savedInstanceState);
     String str =   savedInstanceState.getString("key");
     Log.i("axe.mg","onRestoreInstanceState");Log.i("axe.mg","get value:"+str);
   }

   @Override
   protected void onCreate(Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
     setContentView(R.layout.test_layout);
     Log.i("axe.mg","onCreate()");
   }

   @Override
   protected void onRestart() {
     super.onRestart();
     Log.i("axe.mg","onRestart");
   }

   @Override
   protected void onStart() {
     super.onStart();
     Log.i("axe.mg","onStart()");
   }

   @Override
   protected void onResume() {
     super.onResume();
     Log.i("axe.mg","onResume()");
   }

   @Override
   protected void onPause() {
     super.onPause();
     Log.i("axe.mg","onPause()");
   }

   @Override
   protected void onStop() {
     super.onStop();
     Log.i("axe.mg","onStop()");
   }

   @Override
   protected void onDestroy() {
     super.onDestroy();
     Log.i("axe.mg","onDestroy()");
   }

橫豎屏后獲取到的Log。

09-29 00:26:25.439 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onPause()
09-29 00:26:25.439 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onSaveInstanceState
09-29 00:26:25.439 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onStop()
09-29 00:26:25.439 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onDestroy()

09-29 00:26:25.499 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onCreate()
09-29 00:26:25.499 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onStart()
09-29 00:26:25.499 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onRestoreInstanceState
09-29 00:26:25.499 27568-27568/com.mg.axe.androiddevelop I/axe.mg: get value:value_string
09-29 00:26:25.499 27568-27568/com.mg.axe.androiddevelop I/axe.mg: onResume()

通過Log來分析:

1、Activity銷毀,調用onSaveInstanceState,在onSaveInstanceState方法里面

通過outState.putString("key","value_string");保存了“value_string”這個值。

2、Activity重建,調用onRestoreInstanceState,在onSaveInstanceState方法里面

通過savedInstanceState.getString("key");獲取出了“value_string”這個值。

3、如何防止橫豎屏切換時Activity重建

可能在開發中并不希望橫豎屏切換后Activity重建。此時需要配置configChange參數,

可以設置: android:configChanges="orientation|screenSize"

<activity
   android:name=".develop.ActivityA"
   android:label="@string/app_name"
   android:configChanges="orientation|screenSize"
   android:theme="@style/AppTheme.NoActionBar">
     <intent-filter>
       <action android:name="android.intent.action.MAIN" />
       <category android:name="android.intent.category.LAUNCHER" />
   </intent-filter>
</activity>

配置添加android:configChanges="orientation|screenSize"

(跟蹤framework層代碼,是由于google在android3.2中添加了screensize改變的通知,在轉屏的時候,不僅是orientation發生了改變,screensize同樣也發生了改變所以要添加“screenSize”)

這種情況下橫豎屏切換后不再重建Activity。同時會調用如下方法:

@Override
public void onConfigurationChanged(Configuration newConfig) {
 super.onConfigurationChanged(newConfig);
}

通過代碼驗證:

Activity添加android:configChanges="orientation|screenSize"

<activity
   android:name=".develop.ActivityA"
   android:configChanges="orientation|screenSize"
   android:label="@string/app_name"
   android:theme="@style/AppTheme.NoActionBar">
     <intent-filter>
       <action android:name="android.intent.action.MAIN" />
       <category android:name="android.intent.category.LAUNCHER" />
   </intent-filter>
</activity>
public class ActivityA extends Activity {

  @Override
  public void onConfigurationChanged(Configuration newConfig) {
     super.onConfigurationChanged(newConfig);
     Log.i("axe.mg","onConfigurationChanged");
  }

   @Override
   protected void onSaveInstanceState(Bundle outState) {
     super.onSaveInstanceState(outState);
     outState.putString("key","value_string");
     Log.i("axe.mg","onSaveInstanceState");
   }

   @Override
   protected void onRestoreInstanceState(Bundle savedInstanceState) {
     super.onRestoreInstanceState(savedInstanceState);
     String str =   savedInstanceState.getString("key");
     Log.i("axe.mg","onRestoreInstanceState");Log.i("axe.mg","get value:"+str);
   }

   @Override
   protected void onCreate(Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
     setContentView(R.layout.test_layout);
     Log.i("axe.mg","onCreate()");
   }

   @Override
   protected void onRestart() {
     super.onRestart();
     Log.i("axe.mg","onRestart");
   }

   @Override
   protected void onStart() {
     super.onStart();
     Log.i("axe.mg","onStart()");
   }

   @Override
   protected void onResume() {
     super.onResume();
     Log.i("axe.mg","onResume()");
   }

   @Override
   protected void onPause() {
     super.onPause();
     Log.i("axe.mg","onPause()");
   }

   @Override
   protected void onStop() {
     super.onStop();
     Log.i("axe.mg","onStop()");
   }

   @Override
   protected void onDestroy() {
     super.onDestroy();
     Log.i("axe.mg","onDestroy()");
   }

橫豎平切換后:

09-29 00:54:00.849 21338-21338/com.mg.axe.androiddevelop I/axe.mg: onConfigurationChanged

通過Log分析:

此時Activity不再重建, 不會調用生命周期的方法,也不會調用onSaveInstanceState和onRestoreInstanceState。會調用onConfigurationChanged方法。

+++++++++附件信息:configChanges屬性的值+++++++++++

通過設置這個屬性可以使Activity捕捉設備狀態變化,以下是可以被識別的內容:

設置方法:將下列字段用“|”符號分隔開,例如:“locale|navigation|orientation

"mcc" 國際移動用戶識別碼所屬國家代號是改變了----- sim被偵測到了,去更新mcc mcc是移動用戶所屬國家代號

"mnc" 國際移動用戶識別碼的移動網號碼是改變了------ sim被偵測到了,去更新mnc MNC是移動網號碼,最多由兩位數字組成,用于識別移動用戶所歸屬的移動通信網

"locale" 地址改變了-----用戶選擇了一個新的語言會顯示出來

"touchscreen" 觸摸屏是改變了------通常是不會發生的

"keyboard" 鍵盤發生了改變----例如用戶用了外部的鍵盤

"keyboardHidden" 鍵盤的可用性發生了改變

"navigation" 導航發生了變化-----通常也不會發生

"screenLayout" 屏幕的顯示發生了變化------不同的顯示被激活

"fontScale" 字體比例發生了變化----選擇了不同的全局字體

"uiMode" 用戶的模式發生了變化

"orientation" 屏幕方向改變了

"screenSize" 屏幕大小改變了

"smallestScreenSize" 屏幕的物理大小改變了,如:連接到一個外部的屏幕上

 

 

來自:http://www.jianshu.com/p/f62aed417809

 

 本文由用戶 hocyber 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!