簡(jiǎn)介
在配置Activity時(shí),可以指定android:launchMode
屬性,該屬性用于配置該Activity
的加載模式,支持如下四種模式
- standard 標(biāo)準(zhǔn)模式,這是默認(rèn)的加載模式
- singleTop Task頂單例模式
- singleTask Task內(nèi)單例模式
- singleInstance 全局單例模式
Android采用Task來(lái)管理多個(gè)Activity,當(dāng)啟動(dòng)一個(gè)應(yīng)用時(shí),系統(tǒng)為之創(chuàng)建一個(gè)Task,然后啟動(dòng)入口Activity。Android并沒有為Task提供API,所以我們不能直接訪問Task,只能調(diào)用getTaskID()
來(lái)獲取Activity所在Task的ID。我們可以把Task理解為棧,先啟動(dòng)的Acitivity放在棧底,后啟動(dòng)的Activity放在棧頂。
Activity的加載模式,就是負(fù)責(zé)管理實(shí)例化、加載Activity的方式,并控制Activity與Task之間的加載關(guān)系。
standard 模式
每次通過這種模式啟動(dòng)目標(biāo)Activity時(shí),系統(tǒng)會(huì)為目標(biāo)Activity創(chuàng)建新的實(shí)例,并將該Activity加入當(dāng)前Task棧中。這種模式不會(huì)啟動(dòng)新的Task,新的Acitivty將被加入到原有Task中。
singleTop 模式
將要被啟動(dòng)的Avtivity沒有在Task棧頂時(shí),與standard模式無(wú)異。當(dāng)要啟動(dòng)的Activity已經(jīng)位于Task棧頂時(shí),系統(tǒng)不會(huì)重新創(chuàng)建目標(biāo)Activity實(shí)例,而是復(fù)用已有的Activity實(shí)例。
singleTask 模式
當(dāng)采用這種模式時(shí),要加載的Activity在同一Task內(nèi)只會(huì)存在唯一實(shí)例。系統(tǒng)采用singleTask模式啟動(dòng)目標(biāo)Activity時(shí),會(huì)有如下三種可能性
- 將要啟動(dòng)的Activity在Task棧中不存在,則系統(tǒng)會(huì)創(chuàng)建目標(biāo)Activity的實(shí)例,并將它加入Task棧頂。
- 將要啟動(dòng)的Activity已在Task棧頂,此時(shí)與singleTop模式行為相同。
- 將要啟動(dòng)的目標(biāo)Activity已存在,但不在Task棧頂,系統(tǒng)將會(huì)把位于該Activity上面的所有Activity移出Task棧,從而使目標(biāo)Activity轉(zhuǎn)到棧頂。
singleInstance 模式
在這種模式下,系統(tǒng)保證無(wú)論從哪個(gè)Task棧中啟動(dòng)目標(biāo)Activity,只會(huì)創(chuàng)建一個(gè)目標(biāo)Activity實(shí)例,并會(huì)使用一個(gè)全新的棧來(lái)裝在該Activity實(shí)例。有如下兩種情況
- 要啟動(dòng)的目標(biāo)Activity不存在時(shí),系統(tǒng)會(huì)創(chuàng)建一個(gè)全新的Task,再創(chuàng)建目標(biāo)Activity的實(shí)例,并將它加入新創(chuàng)建的Task的棧頂。
- 要啟動(dòng)的目標(biāo)Activity已存在時(shí),無(wú)論它在哪個(gè)應(yīng)用程序中,無(wú)論它位于哪個(gè)Task棧中,系統(tǒng)會(huì)把該Activity所在的Task轉(zhuǎn)到前臺(tái),從而使該Activity顯示出來(lái)。
采用singleInstance模式啟動(dòng)的Activity,它所在的Task只包含該Acitivty,并且該Activity總位于棧頂(顯而易見)。
簡(jiǎn)單的例子
舉個(gè)例子說(shuō)明一下singleInstance模式。
程序A有三個(gè)界面,加載關(guān)系為A1->A2->A3。其中A2的啟動(dòng)模式為singleInstance
。下面的A2在manifest
中的配置。
<activity
android:name=".SingleInstanceActivity"
android:exported="true"
android:launchMode="singleInstance">
<intent-filter>
<action android:name="test.launch_mode" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
需要說(shuō)明的是設(shè)置android:exported="true"
是為了使外部應(yīng)用可以啟動(dòng)A2。因此要同時(shí)添加一個(gè)intent-filter
,設(shè)置action
,并且一定要加上category android:name="android.intent.category.DEFAULT"
這一條,否則在其他程序啟動(dòng)A2時(shí)會(huì)報(bào)錯(cuò)。
程序B有一個(gè)界面B1,但B1可以啟動(dòng)A2,代碼為
Intent intent = new Intent("test.launch_mode");
startActivity(intent);
打開A程序,依次啟動(dòng)A1,A2,A3,得到的Task棧如圖。
可以看到A2自己位于一個(gè)全新的棧中,通過A2啟動(dòng)的A3則會(huì)放到原來(lái)的棧中。
這時(shí)按下HOME鍵,把A程序切換至后臺(tái),啟動(dòng)B程序,并通過B程序啟動(dòng)A2,A2再啟動(dòng)A3,得到的Task棧如圖
紅色的箭頭是B程序Activity的切換或創(chuàng)建的順序。
可以看到B1啟動(dòng)A2時(shí),由于系統(tǒng)中存在A2,所以僅僅是將A2所在Task切換至前臺(tái),顯示A2。
A2啟動(dòng)A3時(shí),由于A3的啟動(dòng)方式為standard
,所以會(huì)在A3所在程序(A程序)的棧中再實(shí)例化一個(gè)A3。
當(dāng)按BACK依次關(guān)閉Activity時(shí),關(guān)閉順序?yàn)?br>
A3(上)->A3(下)->A1->A2->B1
不同啟動(dòng)模式的使用場(chǎng)景
使用場(chǎng)景的內(nèi)容摘自網(wǎng)上,感覺說(shuō)得還不錯(cuò)。更多的場(chǎng)景,還是在開發(fā)過程中慢慢體會(huì)吧。
singleTop適合接收通知啟動(dòng)的內(nèi)容顯示頁(yè)面
例如,某個(gè)新聞客戶端的新聞內(nèi)容頁(yè)面,如果收到10個(gè)新聞推送,每次都打開一個(gè)新聞內(nèi)容頁(yè)面是很煩人的。singleTask適合作為程序入口點(diǎn)
例如,瀏覽器的主界面。不管從多少個(gè)應(yīng)用啟動(dòng)瀏覽器,只會(huì)啟動(dòng)主界面一次,其余情況都會(huì)走onNewIntent,并且會(huì)清空主界面上面的其他頁(yè)面。singleInstance適合需要與程序分離開的頁(yè)面
例如,鬧鈴提醒,將鬧鈴提醒與鬧鈴設(shè)置分離。singleInstance不要用于中間頁(yè)面,如果用于中間頁(yè)面,跳轉(zhuǎn)會(huì)有問題。