Android如何安全退出多個Activity
在做Android App的時候,幾乎每個App都有一個“退出應用”的功能,如何做到完全退出一個App呢?Google了一些文章,都是都是通過List緩存Activity,在需要退出app的時候,迭代activity列表并調用finish方法,這種方案可以達到完全退出App的功能,但是潛在的問題很大,由于Activity被緩存導致不能被釋放,容易導致OOM,因此這種方式需要慎用。
下面介紹一種解決方案,利用Activity啟動時的flag屬性來解決此問題,其性能優于通過緩存Activity來完全退出App的方案,只是這種解決方案需要滿足一些條件(見分析過程)。
實現步驟:
1:創建一個“空”的輔助Activity,在其onCreate方法中調用finish方法。如下:
- public class LastActivity extends Activity {
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- finish();
- }
- }
2:點擊“退出App”的按鈕執行以下方法:
- findViewById(R.id.quit).setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- Intent intent = new Intent(SecondActivity.this,LastActivity.class);
- intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
- intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- SecondActivity.this.startActivity(intent);
- finish();
- }
- });
通過這兩個步驟可以實現退出整個應用。下面做一個簡單地分析:
測試中使用三個Activity,其執行順序為MainActivity -> SecondActivity -> LastActivity,在SecondActivity中,我們期望點擊“退出app”按鈕來實現完全退出App的功能,此時我們將借助一個輔助Activity(LastActivity)。
首先在觸發點擊事件后,在onClick方法中執行啟動LastActivity操作。在執行點擊事件前我們先查看任務棧中運行著的Activity,可以通過如下的命令查看:
adb shell dumpsys activity
其結果如下:
此時我們可以知道MainActivity跟SecondActivity屬于同一個任務棧,且啟動順序為MainActivity -> SecondActivity。在onClick方法中我們準備啟動LastActivity的時候,其Flag被設置為Intent.FLAG_ACTIVITY_CLEAR_TASK,該flag在android 文檔中的表述為“If set in an Intent passed to Context.startActivity(), this flag will cause any existing task that would be associated with the activity to be cleared before the activity is started.”,其表達的意思為在啟動新的Activity的時候,將清理跟此Activity“相關”的任務棧,此時除了棧中本身的Activity(SecondActivity)都將被清理,然后再啟動Activity(LastActivity)。下面來驗證其是否正確,在LastActivity的onCreate調用finish前加上斷點,然后點擊“退出App”按鈕,查看任務棧的信息如下:
此時我們可以看出,在LastActivity被啟動后,MainActivity被清理了,此時任務棧中只剩下LastActivity跟SecondActivity。接著運行LastActivity的finish方法,LastActivity退出,然后SecondActivty也退出,此時就退出App了。
上面的方式是所有的Activity都存在同一個任務棧中的情況,對于擁有不同任務棧的不同Activity是否也能真正意義上的完全退出App呢?下面我們將在AndroidManifest.xml文件中對SecondActivity跟LastActivity中做如下設置,使其位于不同的任務棧中:
- <activity
- android:name=".SecondActivity"
- android:launchMode="singleTask"
- android:taskAffinity="com.umeng.social"
- />
- <activity
- android:name=".LastActivity"
- android:launchMode="singleTask"
- android:taskAffinity="com.umeng.social.test"
- />
此時我們查看任務棧的情況,如下:
此時我們可以看見MainActivity跟SecondActivity在不同的任務棧中,此時點擊退出App按鈕后,其Activity棧的情況如下:
可以看到此時三個Activity都在不同的任務棧中。細心的同學可能已經發現問題了,為什么LastActivity已經啟動,而MainActivity還沒有被銷毀?是的,這里就是上面提到的'清理跟此Activity“相關”的任務棧'中“相關”兩字的特殊意義了,這說明在清理Activity時是有條件的,其條件為“被clean的Activity必須跟需要啟動的Activity在同一任務棧中”。由于現在MainActivity、SecondActivity、LastActivity在不同任務棧中,因此Activity不會被清理。執行斷點后續代碼,此時Activity棧中依然存在MainActivity,導致整個App是不能完全退出的。
因此采用設置falg的方式完全退出App是有條件的,條件為:”整個App的Activity存在同一個任務棧中“或者任務棧模型滿足以下條件:
目前對于擁有多個任務棧的情況完全退出App的情況暫時沒有想到比較***的解決方案,如果有小伙伴知道,請記得留言,謝謝。