?新建項目ActivityLifeCycleTest,創建主活動后,再新建兩個子活動--NormalActivity和DialogActivity。
現在活動及其對應布局文件創建完畢。
編輯normal_layout.xml文件,代碼如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical" android:layout_width="match_parent"android:layout_height="match_parent"> <TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content" android:text="This is a normal activity"/>
</LinearLayout>
只是簡單是使用了一個TextView,用來顯示一行文字。
然后編輯dialog_layout.xml,代碼如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical" android:layout_width="match_parent"android:layout_height="match_parent"><TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content" android:text="This is a dialog activity"/>
</LinearLayout>
從名字上可以看出來這兩個子活動一個是普通的活動,一個是對話框式的活動。現在我們來修改這個對話框式的活動,打開AndroidManifest.xml文件的<activity>標簽,并修改。代碼如下:
<activity android:name=".DialogActivity"android:theme="@style/Theme.AppCompat.Dialog">
</activity>
這里是活動的注冊碼,并且只給DialogActivity的代碼進行了修改,給它使用了android:theme屬性,這是用來給當前活動指定主題的,Android系統內置了很多主題可以選擇,這里是讓DialogActivity使用對話框式的主題。
接下來,重新定制主活動的布局,將里面的代碼替換成如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/start_normal_activity"android:layout_width="match_parent"android:layout_height="wrap_content" android:text = "Start NormalActivity"/><Buttonandroid:id="@+id/start_dialog_activity"android:layout_width="match_parent"android:layout_height="wrap_content" android:text= "Start DialogActivity"/>
</LinearLayout>
只是添加了兩個按鈕,分別對應兩個子活動。
最后修改MainActivity中的代碼,如下所示:
package com.example.activitylifecycletest;
import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
public class MainActivity extends AppCompatActivity { public static final String TAG = "MainActivity";?//定義當前類名為常量@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button StartNormalActivity = (Button) findViewById(R.id.start_normal_activity);Button StartDialogActivity = (Button) findViewById(R.id.start_dialog_activity);StartNormalActivity.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Intent intent = new Intent(MainActivity.this,NormalActivity.class);startActivity(intent);}});StartDialogActivity.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Intent intent = new Intent(MainActivity.this,DialogActivity.class);startActivity(intent); }})} @Overrideprotected void onStart() {super.onStart();Log.d(TAG,"onStart");}@Overrideprotected void onResume() {super.onResume();Log.d(TAG,"onResume");}@Overrideprotected void onPause() {super.onPause();Log.d(TAG,"onPause");}@Overrideprotected void onStop() {super.onStop();Log.d(TAG,"onStop");}@Overrideprotected void onDestroy() {super.onDestroy();Log.d(TAG,"onDestroy");}
}
在onCreate()方法中,我們分別為兩個按鈕定義了點擊事件,點擊按鈕會啟動相應的Activity,然后會在Activity的7個回調方法中分別打印了一句話,這樣更好的觀察理解活動的生命周期。
- 倘若活動被回收了怎么辦?
假設有一個活動A,在活動A的基礎上啟動了活動B,那么活動A進入了Stop狀態,這時突然由于系統內存不存,將活動A回收了,然后用戶按下Back鍵返回活動A,會出現什么情況呢?還是會正常啟動A的,只不過并不是執行onReStart()方法,而是會執行活動A的onCreate()方法,因為活動A在這種情況會被重新創建一次。那么加入活動A本來是存在臨時數據和狀態的,那么重新創建一次,什么都不沒有顯示了,這不是令人很難過嗎?
經查閱文檔。Activity中還提供了一個onSaveInstanceState()回調方法,這個方法可以保證活動在回收之前一定被調用,因此這個方法來解決活動被回收時,臨時數據得不到保存的問題。
onSaveInstanceState()方法會攜帶一個Bundle類型的參數,Bundle提供了一系列的方法用于保存數據,例如可以使用putString()來保存字符串,使用putInt()保存整型,每個保存方法有兩個參數,一個是鍵,用于后面從Bundle中取值,第二個參數是真正要保存的內容。這么看來,鍵和保存的內容是一一對應的吧。
在MainActivity中添加如下代碼:
@Overrideprotected void onSaveInstanceState(Bundle outState) {super.onSaveInstanceState(outState);String tempData = "Something you just typed";outState.putString("data_key",tempData);}
我們是在MainActivity文件中重寫了這個方法。那么數據是保存好了,但是我們應該在哪里恢復呢?我們之前一直使用的onCreate()方法其實也有一個Bundle類型的參數。這個參數一般情況是null,但是如果在活動被回收之前有通過onSaveInstanceState()方法來保存數據的話,這個參數就會帶有之前所保存的全部數據,我們只需要修改MainActivity的onCreate()方法,代碼如下:
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);Log.d(TAG,"onCreate");setContentView(R.layout.activity_main);if(savedInstanceState !=null){String tempData = savedInstanceState.getString("data_key");Log.d(TAG,tempData);}Button StartNormalActivity = (Button) findViewById(R.id.start_normal_activity);Button StartDialogActivity = (Button) findViewById(R.id.start_dialog_activity);StartNormalActivity.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Intent intent = new Intent(MainActivity.this,NormalActivity.class);startActivity(intent);}});StartDialogActivity.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Intent intent = new Intent(MainActivity.this,DialogActivity.class);startActivity(intent);}});}
Intent還可以結合Bundle一起用于傳輸數據,首先可以把需要傳遞的數據都保存在Bundle對象中,然后再將Buddle對象存放在Intent里。到了目標活動之后先從Intent中取出Bundle,在從Bundle中一一取出數據。