一片林、介紹啟動(dòng)模式之前柄慰,先來(lái)看一下任務(wù)棧:
1.程序打開(kāi)時(shí)就創(chuàng)建了一個(gè)任務(wù)棧, 用于存儲(chǔ)當(dāng)前程序的activity,所有的activity屬于一個(gè)任務(wù)棧踢械。
2.一個(gè)任務(wù)棧包含了一個(gè)activity的集合, 去有序的選擇哪一個(gè)activity和用戶進(jìn)行交互:只有在任務(wù)棧棧頂?shù)腶ctivity才可以跟用戶進(jìn)行交互奸晴。
3.任務(wù)椉妫可以移動(dòng)到后臺(tái), 并且保留了每一個(gè)activity的狀態(tài). 并且有序的給用戶列出它們的任務(wù), 而且還不丟失它們狀態(tài)信息监署。
4.退出應(yīng)用程序時(shí):當(dāng)把所有的任務(wù)棧中所有的activity清除出棧時(shí),任務(wù)棧會(huì)被銷(xiāo)毀,程序退出。
任務(wù)棧的缺點(diǎn):
1.每開(kāi)啟一次頁(yè)面都會(huì)在任務(wù)棧中添加一個(gè)Activity,而只有任務(wù)棧中的Activity全部清除出棧時(shí)纽哥,任務(wù)棧被銷(xiāo)毀钠乏,程序才會(huì)退出,這樣就造成了用戶體驗(yàn)差, 需要點(diǎn)擊多次返回才可以把程序退出了。
2.每開(kāi)啟一次頁(yè)面都會(huì)在任務(wù)棧中添加一個(gè)Activity還會(huì)造成數(shù)據(jù)冗余, 重復(fù)數(shù)據(jù)太多, 會(huì)導(dǎo)致內(nèi)存溢出的問(wèn)題(OOM)春塌。
為了解決任務(wù)棧的缺點(diǎn)晓避,我們引入了啟動(dòng)模式。
啟動(dòng)模式(launchMode)在多個(gè)Activity跳轉(zhuǎn)的過(guò)程中扮演著重要的角色只壳,它可以決定是否生成新的Activity實(shí)例俏拱,是否重用已存在的Activity實(shí)例,是否和其他Activity實(shí)例公用一個(gè)task里吼句。這里簡(jiǎn)單介紹一下task的概念锅必,task是一個(gè)具有棧結(jié)構(gòu)的對(duì)象,一個(gè)task可以管理多個(gè)Activity惕艳,啟動(dòng)一個(gè)應(yīng)用况毅,也就創(chuàng)建一個(gè)與之對(duì)應(yīng)的task。
Activity一共有以下四種launchMode:
- standard
- singleTop
- singleTask
- singleInstance
如何配置Activity的啟動(dòng)模式?
直接在AndroidManifest.xml配置的android:launchMode屬性為以上四種之一即可,代碼如下:
<activity
android:name=".view.activity.LoginActivity"
android:configChanges="orientation|keyboardHidden"
android:launchMode="standard" />
下面我們一一介紹以下四種啟動(dòng)模式及其應(yīng)用場(chǎng)景:
1尔艇、standard
standard模式是默認(rèn)的啟動(dòng)模式尔许,不用為配置android:launchMode屬性即可,當(dāng)然也可以指定值為standard终娃。
stardard模式的原理如下:
如圖所示味廊,每次跳轉(zhuǎn)系統(tǒng)都會(huì)在task中生成一個(gè)新的MainActivity實(shí)例,并且放于棧結(jié)構(gòu)的頂部棠耕,當(dāng)我們按下后退鍵時(shí)余佛,才能看到原來(lái)的MainActivity實(shí)例。
這就是standard啟動(dòng)模式窍荧,不管有沒(méi)有已存在的實(shí)例辉巡,都生成新的實(shí)例放入任務(wù)棧中。
2蕊退、singleTop
我們?cè)谏厦娴幕A(chǔ)上為指定屬性android:launchMode=”singleTop”郊楣,系統(tǒng)就會(huì)按照singleTop啟動(dòng)模式處理跳轉(zhuǎn)行為憔恳。我們重復(fù)上面幾個(gè)動(dòng)作,將會(huì)出現(xiàn)下面的現(xiàn)象:
正如上圖所示净蚤,跳轉(zhuǎn)時(shí)系統(tǒng)會(huì)先在棧結(jié)構(gòu)中尋找是否有一個(gè)MainActivity實(shí)例正位于棧頂钥组,如果有則不再生成新的,而是直接使用今瀑。值得指出的是程梦,如果棧結(jié)構(gòu)中有MainActivity實(shí)例,但是不處于棧頂橘荠,系統(tǒng)還是會(huì)生成新的MainActivity實(shí)例放到棧頂屿附。
3、singleTask
我們修改MainActivity的屬性android:launchMode=”singleTask”哥童,演示結(jié)果如下:
可以發(fā)現(xiàn)當(dāng)SecondActivity跳轉(zhuǎn)到FirstActivity時(shí)SecondActivity消失了拿撩。沒(méi)錯(cuò),在這個(gè)跳轉(zhuǎn)過(guò)程中系統(tǒng)發(fā)現(xiàn)有存在的FirstActivity實(shí)例如蚜,于是不再生成新的實(shí)例压恒,而是將FirstActivity之上的Activity實(shí)例統(tǒng)統(tǒng)出棧,將FirstActivity變?yōu)闂m攲?duì)象错邦,顯示到幕前探赫。
4、singleInstance
這種啟動(dòng)模式比較特殊撬呢,因?yàn)樗鼤?huì)啟用一個(gè)新的棧結(jié)構(gòu)伦吠,將Activity放置于這個(gè)新的棧結(jié)構(gòu)中,并保證不再有其他Activity實(shí)例進(jìn)入魂拦。
我們修改MainActivity的launchMode=”standard”毛仪,SecondActivity的launchMode=”singleInstance”,由于涉及到了多個(gè)棧結(jié)構(gòu)芯勘,我們需要在每個(gè)Activity中顯示當(dāng)前棧結(jié)構(gòu)的id箱靴,所以我們?yōu)槊總€(gè)Activity添加如下代碼:
TextView textView = (TextView) findViewById(R.id.tv);
textView.setText("current task id:" + this.getTaskId());
效果圖:
1.我們看到從MainActivity跳轉(zhuǎn)到SecondActivity時(shí),重新啟用了一個(gè)新的棧結(jié)構(gòu)荷愕,來(lái)放置SecondActivity實(shí)例衡怀,然后按下后退鍵,再次回到原始棧結(jié)構(gòu)安疗,圖中上半部分抛杨;
2.圖中下半部分顯示的在SecondActivity中再次跳轉(zhuǎn)到MainActivity,這個(gè)時(shí)候系統(tǒng)會(huì)在原始棧結(jié)構(gòu)中生成一個(gè)MainActivity實(shí)例荐类,然后回退兩次怖现,注意,并沒(méi)有退出玉罐,而是回到了SecondActivity屈嗤,為什么呢潘拨?是因?yàn)閺腟econdActivity跳轉(zhuǎn)到MainActivity的時(shí)候,我們的起點(diǎn)變成了SecondActivity實(shí)例所在的棧結(jié)構(gòu)恢共,這樣一來(lái),我們需要“回歸”到這個(gè)棧結(jié)構(gòu)璧亚。
解釋:第一次back關(guān)閉了由SecondActivity啟動(dòng)的MainActivity此時(shí)因?yàn)樘幱诘谝粋€(gè)棧中讨韭,所以顯示的是第一個(gè)MainActivity,再次back這時(shí)第一個(gè)MainActivity消失癣蟋,同時(shí)第一個(gè)棧被銷(xiāo)毀透硝,SecondActivity展示到前臺(tái)。
應(yīng)用場(chǎng)景:
singleTop適合接收通知啟動(dòng)的內(nèi)容顯示頁(yè)面疯搅。例如濒生,某個(gè)新聞客戶端的新聞內(nèi)容頁(yè)面,如果收到10個(gè)新聞推送幔欧,每次都打開(kāi)一個(gè)新聞內(nèi)容頁(yè)面是很煩人的罪治。
singleTask適合作為程序入口點(diǎn)。例如瀏覽器的主界面礁蔗。不管從多少個(gè)應(yīng)用啟動(dòng)瀏覽器觉义,只會(huì)啟動(dòng)主界面一次,其余情況都會(huì)走onNewIntent浴井,并且會(huì)清空主界面上面的其他頁(yè)面晒骇。之前打開(kāi)過(guò)的頁(yè)面,打開(kāi)之前的頁(yè)面就ok磺浙,不再新建洪囤。
singleInstance適合需要與程序分離開(kāi)的頁(yè)面。例如鬧鈴提醒撕氧,將鬧鈴提醒與鬧鈴設(shè)置分離瘤缩。singleInstance不要用于中間頁(yè)面,如果用于中間頁(yè)面伦泥,跳轉(zhuǎn)會(huì)有問(wèn)題款咖,比如:A -> B (singleInstance) -> C,完全退出后奄喂,在此啟動(dòng)铐殃,首先打開(kāi)的是B。