Activity的啟動模式以及onNewIntent(Intent intent)


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標記,如下所示:

 

 

Java代碼 復制代碼  收藏代碼
  1. Intent intent = new Intent(this, B.class);    
  2. intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);   
  3. 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,則在上面的代碼里再加上:

 

 

Java代碼 復制代碼  收藏代碼
  1. 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,則可以像下面寫代碼:

 

 

 

Java代碼 復制代碼  收藏代碼
  1. Intent intent = new Intent(this, MainActivity.class);   
  2. intent.addFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT);    
  3. 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 }

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM