Android的啟動模式

  在一個Android應用中,不可避免的會包含多個Activity,當開啟多個Activity之後,當按後退鍵時,一般會回退到上一個Activity,這一點和瀏覽器有點類似,但是需要註意的是Android系統隻能回退到上一個Activity,沒辦法前進到下一個Activity,維護這個功能就需要瞭解到一個回退棧(Back Stack)的概念。
什麼是回退棧?
  首先先來瞭解一下應用程序與進程的關系,眾所周知,Android下有四大組件:Activity、Service、Receiver、ContentProvider。一般開發一個應用程序,會包含多個Android組件,所以應用程序是一組組件的集合,而進程則是運行這些組件的載體。
  而回退棧(Back Stack)隻是針對Activity而言的,它是用來維護用的界面體驗的,使一個Task讓用戶感覺就是一個應用,而無論其中的Activity是否來自同一個應用程序,所以不要把回退棧和進程弄混瞭。設備的Home頁面是大多數Task的起始位置,當用戶點擊一個應用程序圖標的時候,應用的Task就會來到前臺,並把應用的主Activity壓入BackStack的棧頂,並獲得焦點,這個Activity稱為根Activity,而在BackStack中的Activity可以通過點擊回退鍵彈出棧並銷毀,這時就會使上一個Activity獲得焦點,直到用戶返回到Home頁,而當BackStack中的Activity都被彈出銷毀之後,這個Task就不復存在瞭,但是這個程序的進程還存在(不在此時銷毀)。
Task的狀態
  就像上面介紹的,每個Task都存在一個BackStack,而系統中可以存在多個Task,但是每次隻有一個Task獲得前臺焦點,一般而言,系統允許用戶在多個Task中切換,而被至於後臺的Task中的Activity,將被置於Stopped狀態。實際上,同一個Task中的Activity,隻要不存在於棧頂並且獲得前臺焦點的Activity,那麼它就是一個Stopped的狀態。下
Activity的啟動模式
  根據Activity的不同的啟動模式,它在BackStack中的狀態是不一樣的。Activity可以通過AndroidManifest.xml清單文件配置,在節點中的android:launchMode屬性設置。它有四個選項:
· standard
· singleTop
· singleTask
· singleInstance
standard:
  標準啟動模式,也是默認啟動模式,如果不設置android:launchMode屬性的話。standard模式下的Activity會依照啟動的順序壓入BackStack中。
  下圖是standard模式下,Activity的壓棧和回退操作示意圖:

singleTop:
  單頂模式,這種Activity啟動模式,啟動一個Activity的時候如果發現BackStack的棧頂已經存在這個Activity瞭,就不會去重新創建新的Activity,而是復用這個棧頂已經存在的Activity,避免同一個Activity被重復開啟。

  下圖是singleTZ喎?/kf/ware/vc/” target=”_blank” class=”keylink”>vcMSjyr3PwqOsQWN0aXZpdHm1xNG51bu6zbvYzcuy2df3yr7S4s28o7o8L3A+CjxwPjxpbWcgc3JjPQ==”/uploadfile/Collfiles/20140310/20140310140210338.jpg” alt=”\”>

  singleTop的應用場景很多,一般適用於可以復用而又有多個開啟渠道的Activity,避免當一個Activity已經開啟並獲得焦點後,再次重復開啟。比如說Android系統瀏覽器的書簽頁面,就是一個singleTop模式的Activity。Android的瀏覽器是基於WebKit內核編寫的,它是支持JavaScript腳本語言的,可以通過JavaScript腳本設置瀏覽器書簽,這樣如果存在多個頁面存在保存書簽的JavaScript腳本,就會導致書簽頁面被多次開啟,所以書簽頁面被設置為singleTop模式,這樣可以避免在保存多個書簽的時候重復開啟書簽頁面。
singleTask:
  開啟一個Activity的時候,檢查BackStack裡面是否有這個Activity的實例存在,如果存在的話,結束BackStack裡這個Activity上所有的其他Activity。

  下圖是singleTask模式下,Activity的壓棧和回退操作示意圖:

  singleTask的的適用場景為一般程序的主頁面,當回退到主頁面的時候,清除BackStack中,它之上的所有Activity,這樣避免程序導航邏輯的混亂。比如Android系統的瀏覽器的主頁面,就是singleTask模式的,上面提到,android下瀏覽器是Webkit內核的,它是由C語言編寫的,而每次打開新的網頁如果重新開啟一個Activity,是非常耗費系統資源的(需要解析HTML、Script腳本),所以被設置為singleTask模式,這樣在瀏覽器應用裡,無論打開多少個頁面,使用的都是同一個Activity。所以以後如果存在很耗費系統資源的Activity,可以考慮使用singleTask開啟模式。
singleInstance:
  被標記為singleInstance啟動模式的Activity,在啟動的時候,會開啟一個新的BackStack,這個BackStack裡隻有一個Activity的實例存在,並且把這個BackStack獲得焦點。這是一種很極端的模式,它會導致整個設備的操作系統裡,隻會存在一個這個Activity示例,無論是從何處被啟動的。

  下圖是singleInstance模式下,Activity的壓棧和回退操作示意圖: 

  singleInstance一般適用於需要在系統中隻存在一個實例的場景,比如Android系統的來電頁面,多次來電均使用的是一個Activity。
  當然,在Android中,除瞭在AndroidManifest.xml清單文件中配置LauncherMode屬性外,還可以在代碼中設置啟動模式。在組件中,啟動一個Activity,需要用到startActivity()方法,其中傳遞一個Intent,可以使用Intent.setFlags(int flags)來設置新啟動的Activity的啟動模式,而通過代碼設置Activity的啟動模式的方式,優先級要高於在AndroidManifest.xml清單文件中的配置。
  Intent.setFlag(int flags)方法傳遞的一個整形的數據,被Android系統設置為瞭常量:
· FLAG_ACTIVITY_NEW_TASK:這個標識會使新啟動的Activity獨立創建一個Task。
· FLAG_ACTIVITY_CLEAR_TOP:這個標識會使新啟動的Activity檢查是否存在於Task中,如果存在則清除其之上的Activity,使它獲得焦點,並不重新實例化一個Activity,一般結合FLAG_ACTIVITY_NEW_TASK一起使用。

· FLAG_ACTIVITY_SINGLE_TOP:等同於在LauncherMode屬性設置為singleTop。

總結:

standard 每次都會新建,每個Task都可以有,且每個Task都可以有多個實例(每個Task都可以有,且可以有多個)
singleTop 當前實例如果在棧頂,就不新建實例,調用其OnNewIntent。 如不在棧頂,則新建實例 (每個Task都可以有,且可以有多個,在棧頂時可復用,否則創建)
singleTask 新建一個Task,如果已經有其他的Task並且包含該實例,那就直接調用那個Task的實例。(隻有一個Task中會有)
singleInstance 新建一個Task,且在該Task中隻有它的唯一一個實例。 (隻有一個Task會有,且該Task中隻有它)

發佈留言