天天看點

"singleTask"模式 切換到新的棧中

本文截取了網絡資源的結論部分 對singletask 啟動模式做筆記記錄。

雖然SubActivity的launchMode被設定為"singleTask"模式,但是它并不像官方文檔描述的一樣:The system creates a new task and instantiates the activity at the root of the new task,而是在跟它有相同taskAffinity的任務中啟動,并且位于這個任務的堆棧頂端,于是,前面那個圖中,就會出現一個帶着"singleTask"标簽的箭頭指向一個任務堆棧頂端的Activity Y了。

        那麼,我們有沒有辦法讓一個"singleTask"的Activity在新的任務中啟動呢?答案是肯定的。從上面的代碼分析中,隻要我們能夠進入函數startActivityUncheckedLocked的這個if語句中: [java] view plain copy

  1.  if (r.resultTo == null && !addingToTask  
  2.        && (launchFlags&Intent.FLAG_ACTIVITY_NEW_TASK) != 0) {  
  3. // todo: should do better management of integers.  
  4.        mService.mCurTask++;  
  5.        if (mService.mCurTask <= 0) {  
  6.             mService.mCurTask = 1;  
  7.        }  
  8.        r.task = new TaskRecord(mService.mCurTask, r.info, intent,  
  9.                   (r.info.flags&ActivityInfo.FLAG_CLEAR_TASK_ON_LAUNCH) != 0);  
  10.        if (DEBUG_TASKS) Slog.v(TAG, "Starting new activity " + r  
  11.                   + " in new task " + r.task);  
  12.         newTask = true;  
  13.         if (mMainStack) {  
  14.               mService.addRecentTaskLocked(r.task);  
  15.         }  
  16.  }  

        那麼,這個即将要啟動的Activity就會在新的任務中啟動了。進入這個if語句需要滿足三個條件,r.resultTo為null,launchFlags的Intent.FLAG_ACTIVITY_NEW_TASK位為1,并且addingToTask值為false。從上面的分析中可以看到,當即将要啟動的Activity的launchMode為"singleTask",并且調用startActivity時不要求傳回要啟動的Activity的執行結果時,前面兩個條件可以滿足,要滿足第三個條件,隻要目前系統不存在affinity屬性值等于即将要啟動的Activity的taskAffinity屬性值的任務就可以了。

        我們可以稍微修改一下上面的AndroidManifest.xml配置檔案來做一下這個實驗:

[java] view plain copy

  1. <?xml version="1.0" encoding="utf-8"?>    
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"    
  3.     package="shy.luo.task"    
  4.     android:versionCode="1"    
  5.     android:versionName="1.0">    
  6.     <application android:icon="@drawable/icon" android:label="@string/app_name">    
  7.         <activity android:name=".MainActivity"    
  8.                   android:label="@string/app_name"  
  9.                   android:taskAffinity="shy.luo.task.main.activity">    
  10.             <intent-filter>    
  11.                 <action android:name="android.intent.action.MAIN" />    
  12.                 <category android:name="android.intent.category.LAUNCHER" />    
  13.             </intent-filter>    
  14.         </activity>    
  15.         <activity android:name=".SubActivity"    
  16.                   android:label="@string/sub_activity"  
  17.                   android:launchMode="singleTask"  
  18.                   android:taskAffinity="shy.luo.task.sub.activity">    
  19.             <intent-filter>    
  20.                 <action android:name="shy.luo.task.subactivity"/>    
  21.                 <category android:name="android.intent.category.DEFAULT"/>    
  22.             </intent-filter>    
  23.         </activity>    
  24.     </application>    
  25. </manifest>    

        注意,這裡我們設定MainActivity的taskAffinity屬性值為"shy.luo.task.main.activity",設定SubActivity的taskAffinity屬性值為"shy.luo.task.sub.activity"。重新編譯一下程式,在模拟器上把這個應用程式再次跑起來,用“adb shell dumpsys activity”指令再來檢視一下系統運作的的任務,就會看到:

[html] view plain copy

  1. Running activities (most recent first):  
  2.     TaskRecord{4069c020 #4 A shy.luo.task.sub.activity}  
  3.       Run #2: HistoryRecord{40725040 shy.luo.task/.SubActivity}  
  4.     TaskRecord{40695220 #3 A shy.luo.task.main.activity}  
  5.       Run #1: HistoryRecord{406b26b8 shy.luo.task/.MainActivity}  
  6.     TaskRecord{40599c90 #2 A com.android.launcher}  
  7.       Run #0: HistoryRecord{40646628 com.android.launcher/com.android.launcher2.Launcher}  

        這裡就可以看到,SubActivity和MainActivity就分别運作在不同的任務中了。

        至此,我們總結一下,設定了"singleTask"啟動模式的Activity的特點:

        1. 設定了"singleTask"啟動模式的Activity,它在啟動的時候,會先在系統中查找屬性值affinity等于它的屬性值taskAffinity的任務存在;如果存在這樣的任務,它就會在這個任務中啟動,否則就會在新任務中啟動。是以,如果我們想要設定了"singleTask"啟動模式的Activity在新的任務中啟動,就要為它設定一個獨立的taskAffinity屬性值。

        2. 如果設定了"singleTask"啟動模式的Activity不是在新的任務中啟動時,它會在已有的任務中檢視是否已經存在相應的Activity執行個體,如果存在,就會把位于這個Activity執行個體上面的Activity全部結束掉,即最終這個Activity執行個體會位于任務的堆棧頂端中。