Activity的Launch mode詳解 singleTask正解

銳湃發表於2015-11-12

本文參考了此文http://hi.baidu.com/amauri3389/blog/item/a54475c2a4b2f040b219a86a.html

另附 android task與back stack開發文件翻譯

參見:http://mypyg.iteye.com/blog/919643     http://blog.csdn.net/infsafe/article/details/5666964

Activity有四種載入模式:standard(預設), singleTop, singleTask和 singleInstance。以下逐一舉例說明他們的區別:

 

standard:Activity的預設載入方法,即使某個Activity在Task棧中已經存在,另一個activity通過Intent跳轉到該activity,同樣會新建立一個例項壓入棧中。例如:現在棧的情況為:A B C D,在D這個Activity中通過Intent跳轉到D,那麼現在的棧情況為: A B C D D 。此時如果棧頂的D通過Intent跳轉到B,則棧情況為:A B C D D B。此時如果依次按返回鍵,D  D C B A將會依次彈出棧而顯示在介面上。

 

singleTop:如果某個Activity的Launch mode設定成singleTop,那麼當該Activity位於棧頂的時候,再通過Intent跳轉到本身這個Activity,則將不會建立一個新的例項壓入棧中。例如:現在棧的情況為:A B C D。D的Launch mode設定成了singleTop,那麼在D中啟動Intent跳轉到D,那麼將不會新建立一個D的例項壓入棧中,此時棧的情況依然為:A B C D。但是如果此時B的模式也是singleTop,D跳轉到B,那麼則會新建一個B的例項壓入棧中,因為此時B不是位於棧頂,此時棧的情況就變成了:A B C D B。

 

singleTask:如果某個Activity是singleTask模式,那麼Task棧中將會只有一個該Activity的例項。例如:現在棧的情況為:A B C D。B的Launch mode為singleTask,此時D通過Intent跳轉到B,則棧的情況變成了:A B。而C和D被彈出銷燬了,也就是說位於B之上的例項都被銷燬了。

關於singleTask這個網上頗有爭議,google api說singTask模式只能啟動一個task,且總是位於棧底,這個也不是完全正確

分2種情況:

  1、如果在同一個應用(apk)中使用singleTask,剛不在棧底,對應於下面的情況一

  2、如果從不同應用啟動一個singleTask的activity,剛依賴於此activity所在的棧,如果之前沒有執行過,則新建棧處於棧底,如果有執行過,則有可能不在棧底,對應於情況二

     情況一:如果在本程式中啟動singleTask的activity:假設ActivityA是程式的入口,是預設的模式(standard),ActivityB是singleTask 模式,由ActivityA啟動,剛ActivityB不會位於棧底,不是根元素,不會啟動新的task,此種情況ActivityB會和ActivityA在一個棧中,位於ActivityA上面

  情況二:如果ActivityB由另外一個程式啟動:假設apkA是情況一中的應用,apkB是另外一個測試程式,在apkB中啟動apkA中的ActivityB,再分兩種情況,如果ActivityB未啟動過,ActivityB會位於棧底,是根元素,會啟動新的task;如果ActivityB啟動過,則ActivityB保持原來的位置不變,在棧底或者棧頂,移除掉ActivityB之上所有的activity(如果有),見下圖

 

此圖就是儲存了activitY所在的棧的情況,按返回鍵的時候,會首先依次移除掉activitY所在的棧的activity,然後才是activity2的

注意singleTask模式的Activity不管是位於棧頂還是棧底,再次執行這個Activity時,都會destory掉它上面的Activity來保證整個棧中只有一個自己,切記切記

singleInstance:將Activity壓入一個新建的任務棧中。例如:Task棧1的情況為:A B C。C通過Intent跳轉到D,而D的Launch mode為singleInstance,則將會新建一個Task棧2。此時Task棧1的情況還是為:A B C。Task棧2的情況為:D。此時螢幕介面顯示D的內容,如果這時D又通過Intent跳轉到D,則Task棧2中也不會新建一個D的例項,所以兩個棧的情況也不會變化。而如果D跳轉到C,則棧1的情況變成了:A B C C,因為C的Launch mode為standard,此時如果再按返回鍵,則棧1變成:A B C。也就是說現在介面還顯示C的內容,不是D。

好了,現在有一個問題就是這時這種情況下如果使用者點選了Home鍵,則再也回不到D的即時介面了。如果想解決這個問題,可以為D在Manifest.xml檔案中的宣告加上:

 

<intent-filter>

       <action android:name="android.intent.action.MAIN" />

        <category android:name="android.intent.category.LAUNCHER" />

 </intent-filter>

 

加上這段之後,也就是說該程式中有兩個這種宣告,另一個就是那個正常的根activity,在打成apk包安裝之後,在程式列表中能看到兩個圖示,但是如果都執行的話,在工作管理員中其實也只有一個。上面的情況點選D的那個圖示就能回到它的即時介面(比如一個EditText,以前輸入的內容,現在回到之後依然存在)。

 

PS:intent-filter中 <action android:name="android.intent.action.MAIN" />和 <category android:name="android.intent.category.LAUNCHER" />兩個過濾條件缺一不可才會在程式列表中新增一個圖示,圖示下的顯示文字是android:label設定的字串。

測試程式:http://files.cnblogs.com/xiaoQLu/launchModeTest.zip


轉自:http://www.cnblogs.com/xiaoQLu/archive/2011/09/29/2195742.html

相關文章