1. 程式人生 > >Activity的四種啟動模式(回退棧)

Activity的四種啟動模式(回退棧)

 在一個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的狀態。下圖為官方文件中關於Task前後臺的示例圖:

Activity的啟動模式

  根據Activity的不同的啟動模式,它在BackStack中的狀態是不一樣的。Activity可以通過AndroidManifest.xml清單檔案配置,在<Activity />節點中的android:launchMode屬性設定。它有四個選項:

standard

  標準啟動模式,也是預設啟動模式,如果不設定android:launchMode屬性的話。standard模式下的Activity會依照啟動的順序壓入BackStack中。

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

singleTop

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

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

  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。