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.cnblogs.com/-cyb/articles/Android_onConfigurationChanged.html (要添加screenSize的問題)
http://www.cnblogs.com/carlo/p/4311010.html (附件信息:configChanges屬性的值)
參考書籍:Android開發藝術探索。