Activity的啟動模式
啟動模式允許開發者定義一個activity的新實例如何與當前的Task關聯。可以定義使用倆種方法來定義。
如果Activity A開啟Activity B, Activity B就可以在它的manifest文件中定義它與當前的task如何關聯,Activity A也可以要求activity B應該如何與當前的task關聯。如果兩個activity都定義了Activity B應該如何與一個task關聯,Activity A的要求(在intent中定義的)將會覆蓋Activity B中要求(在manifest文件中定義的)。
注意:一些在manifest中的登錄模式在intent中不再可用,同樣地,一些在intent中定義的標志也可能沒有在manifest中未定義。
Using the manifest file
當在manifest文件中聲明activity時,可以指定這個activity開啟時如何與當前task關聯。
<activity>標簽的launchMode屬性可以設置為四種不同的模式:
“standard”(默認模式)
“singleTop”
“singleTask”
“singleInstance”
這幾種模式的區別體現以下四點上:
1)當這個activity被激活的時候,會放入哪個任務棧。
對於“standard”和“singleTop”模式,這個新被激活的activity會放入和之前的activity相同的任務棧中――除非Intent對象包含FLAG_ACTIVITY_NEW_TASK標志。
“singleTask”並不會每次都新啟動一個task。如果已經存在一個task與新activity親和度(taskAffinity)一樣,則activity將啟動到該task。如果不是,才啟動一個新task。同一個application里面,每個activity的taskAffinity默認都是一樣的。
“singleInstance”模式則表示這個新被激活的activity會重新開啟一個任務棧,並作為這個新的任務棧的唯一的activity。
2)是否可以存在這個activity類型的多個實例。
對於“standard”和“singleTop”模式,可以有多個實例,並且這些實例可以屬於不同的任務棧,每個任務棧也可以包含有這個activity類型的多個實例。
“singleTop"要求如果創建intent的時候棧頂已經有要創建 的Activity的實例,則將intent發送給該實例,而不發送給新的實例。
“singleTask”和”singleInstance”則限制只生成一個實例。
3)包含此activity的任務棧是否可以包含其它的activity。
“singleInstance”模式表示包含此activity的任務棧不可以包含其它的activity。若此activity啟動了另一個activity組件,那么無論那個activity組件的啟動模式是什么或是Intent對象中是否包含了FLAG_ACTIVITY_NEW_TASK標志,它都會被放入另外的任務棧。在其它方面“singleInstance”模式和“singleTask”模式是一樣的。
其余三種啟動模式則允許包含此activity的任務棧包含其它的activity。
4)是否每次都生成新實例
對於默認的“standard”模式,每當響應一個Intent對象,都會創建一個這種activity類型的新的實例。即每一個activity實例處理一個intent。
對於“singleTop”模式,只有當這個activity的實例當前處於任務棧的棧頂位置,則它會被重復利用來處理新到達的intent對象。否則就和“standard”模式的行為一樣。
“singleInstance”是其所在棧的唯一activity,它會每次都被重用。
對於“singleTask”模式的acitvity,在其上面可能存在其它的activity組件,所以它的位置並不是棧頂,在這種情況下,intent對象會被丟棄。(雖然會被丟棄,但是這個intent對象會使這個任務棧切換到前台)
注意:
當已經存在的activity實例處理新的intent時候,會調用onNewIntent()方法
若為了處理一個新到達的intent對象而創建了一個activity實例,則用戶按下“BACK”鍵就會退到之前的那個activity。但若這個新到達的intent對象是由一個已經存在的activity組件來處理的,那么用戶按下“BACK” 鍵就不會回退到處理這個新intent對象之前的狀態了。
Using Intent flags
當開啟一個activity時,可以通過在intent中包含標志來修改activity的默認的與當前task的關聯,然后將該intent傳遞給startActivity().可以修改的默認的標志為:
- FLAG_ACTIVITY_NEW_TASK
在一個新的task中開啟一個activity。如果包含該activity的task已經運行,該task就回到前台,activity通過onNewIntent()接受處理該intent。
這是與"singleTask"登錄模式相同的行為。 - FLAG_ACTIVITY_SINGLE_TOP
如果要被開啟的activity是當前的activity(在返回棧的頂部),已經存在的實例通過onNewIntent()接收一個調用,然后處理該intent,而非重新創建一個新的實例。
這與"singleTop"登錄模式有相同的行為。 - FLAG_ACTIVITY_CLEAR_TOP
如果要被開啟的activity已經在當前的task中運行,系統不會生成該activity的一個新的實例,在該棧頂部的所有其他的activity會被銷毀,這個intent通過 onNewIntent()被傳遞給該重新運行的activity的實例(現在在棧頂部)。
manifest中沒有相對應的屬性。
FLAG_ACTIVITY_CLEAR_TOP經常和FLAG_ACTIVITY_NEW_TASK一起使用.當一起使用時,這些標志可以確定一個存在的activity在另一個task中的位置,並且將其放置於可以響應intent的位置(FLAG_ACTIVITY_NEW_TASK確定該activity,然后FLAG_ACTIVITY_CLEAR_TOP銷毀頂部其他的activity)。如果指定的activity的登錄模式是"standard",也會被從棧中移除,一個新的實例也會被登錄到它的位置來處理到來的intent。那是因為當登錄模式為 "standard"時,一個新的實例總是被創建
注意: 其實以上的解釋一起用非常復雜,比如一般系統默認activity是 standard,但當我activity代碼設置為FLAG_ACTIVITY_NEW_TASK時仍然還是生成新的activity,當設置FLAG_ACTIVITY_CLEAR_TOP 時也是生成新的activity,當FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_NEW_TASK時也是生成新的activity,或許這句好是經典“登錄模式為 "standard"時,一個新的實例總是被創建”,以下的兩種方式是我經常用的。
Activity的兩種啟動模式:FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_REORDER_TO_FRONT
1. 如果已經啟動了四個Activity:A,B,C和D。在D Activity里,我們要跳到B Activity,同時希望C finish掉,可以在startActivity(intent)里的intent里添加flags標記,如下所示:
- Intent intent = new Intent(this, B.class);
- intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
- startActivity(intent);
Intent intent = new Intent(this, B.class); intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); startActivity(intent);
這樣啟動B Activity,就會把D,C都finished掉,如果你的B Activity的啟動模式是默認的(multiple) ,則B Activity會finished掉,再啟動一個新的Activity B。
如果不想重新再創建一個新的B Activity,則在上面的代碼里再加上:
- intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
這樣B Activity就會再創建一個新的了,而是會重用之前的B Activity,同時調用B Activity的onNewIntent()方法。
2. 如果已經啟動了四個Activity:A,B,C和D,在D Activity里,想再啟動一個Actvity B,但不變成A,B,C,D,B,而是希望是A,C,D,B,則可以像下面寫代碼:
- Intent intent = new Intent(this, MainActivity.class);
- intent.addFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT);
- startActivity(intent);
Intent intent = new Intent(this, MainActivity.class); intent.addFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT); startActivity(intent);
onNewIntent(Intent intent):
在Android應用程序開發的時候,從一個Activity啟動另一個Activity並傳遞一些數據到新的Activity上非常簡單,但是當您需要讓后台運行的Activity回到前台並傳遞一些數據可能就會存在一點點小問題。
首先,在默認情況下,當您通過Intent啟到一個Activity的時候,就算已經存在一個相同的正在運行的Activity,系統都會創建一個新的Activity實例並顯示出來。為了不讓Activity實例化多次,我們需要通過在AndroidManifest.xml配置activity的加載方式(launchMode)以實現單任務模式,如下所示:
1 <activity android:label="@string/app_name" android:launchmode="singleTask"android:name="Activity1">
2 </activity>
launchMode為singleTask的時候,通過Intent啟到一個Activity,如果系統已經存在一個實例,系統就會將請求發送到這個實例上,但這個時候,系統就不會再調用通常情況下我們處理請求數據的onCreate方法,而是調用onNewIntent方法,如下所示:
1 protected void onNewIntent(Intent intent) {
2 super.onNewIntent(intent);
3 setIntent(intent);//must store the new intent unless getIntent() will return the old one
4 processExtraData();
5 }
不要忘記,系統可能會隨時殺掉后台運行的Activity,如果這一切發生,那么系統就會調用onCreate方法,而不調用onNewIntent方法,一個好的解決方法就是在onCreate和onNewIntent方法中調用同一個處理數據的方法,如下所示:
01 public void onCreate(Bundle savedInstanceState) {
02 super.onCreate(savedInstanceState);
03 setContentView(R.layout.main);
04 processExtraData();
05 }
06
07 protected void onNewIntent(Intent intent) {
08 super.onNewIntent(intent);
09 setIntent(intent);//must store the new intent unless getIntent() will return the old one
10 processExtraData()
11 }
12
13 private void processExtraData(){
14 Intent intent = getIntent();
15 //use the data received here
16 }