最優雅退出 Android 應用程式的 6 種方式

soul_code發表於2016-01-10

我們先來看看幾種常見的退出方法(不優雅的方式)

一、容器式

建立一個全域性容器,把所有的Activity儲存起來,退出時迴圈遍歷finish所有Activity

import java.util.ArrayList;
import java.util.List;

import android.app.Activity;
import android.os.Bundle;

public class BaseActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 新增Activity到堆疊
        AtyContainer.getInstance().addActivity(this);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 結束Activity&從棧中移除該Activity
        AtyContainer.getInstance().removeActivity(this);
    }

}

class AtyContainer {

    private AtyContainer() {
    }

    private static AtyContainer instance = new AtyContainer();
    private static List<Activity> activityStack = new ArrayList<Activity>();

    public static AtyContainer getInstance() {
        return instance;
    }

    public void addActivity(Activity aty) {
        activityStack.add(aty);
    }

    public void removeActivity(Activity aty) {
        activityStack.remove(aty);
    }

    /**
     * 結束所有Activity
     */
    public void finishAllActivity() {
        for (int i = 0, size = activityStack.size(); i < size; i++) {
            if (null != activityStack.get(i)) {
                activityStack.get(i).finish();
            }
        }
        activityStack.clear();
    }

}

這種方法比較簡單, 但是可以看到activityStack持有這Activity的強引用,也就是說當某個Activity異常退出時,activityStack沒有即使釋放掉引用,就會導致記憶體問題,接下來我們看一種類似的方式,但是會稍微優雅一點點

二、廣播式

通過在BaseActivity中註冊一個廣播,當退出時傳送一個廣播,finish退出

public class BaseActivity extends Activity {

    private static final String EXITACTION = "action.exit";

    private ExitReceiver exitReceiver = new ExitReceiver();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        IntentFilter filter = new IntentFilter();
        filter.addAction(EXITACTION);
        registerReceiver(exitReceiver, filter);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        unregisterReceiver(exitReceiver);
    }

    class ExitReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
            BaseActivity.this.finish();
        }

    }

}

三、程式式

通過直接殺死當前應用的程式來結束應用,簡單粗暴,而且有(wu)效!

 android.os.Process.killProcess(android.os.Process.myPid());
    System.exit(0);

    ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
    manager.killBackgroundProcesses(getPackageName());

     這三種都能達到同樣的效果,但是在模擬器上都會彈出 Unfortunately , XXX has stopped 訊息提示框,但確實能退出應用。部分真機直接失效,只能finish當前Activity(比如我手上這臺小米note,國產的幾款ROM fw層改動太多,使用這種方式需慎重) 。

四、 RS優雅式

什麼是RS式呢?即Receiver+singleTask 。我們知道Activity有四種載入模式,而singleTask就是其中的一種,使用這個模式之後,當startActivity時,它先會在當前棧中查詢是否存在Activity的例項,如果存在,則將其至於棧頂,並將其之上的所有Activity移除棧。我們開啟一個app,首先是一個splash頁面,然後會finish掉splash頁面。跳轉到主頁。然後會在主頁進行N次的跳轉,期間會產生數量不定的Activity,有的被銷燬,有的駐留在棧中,但是棧底永遠是我們的HomeActivity。這樣就讓問題變得簡單很多了。我們只需兩步操作即可優雅的實現app的退出。

  • 1、在HomeActivity註冊一個退出廣播,和第二個廣播式一樣,但是這裡只需要在HomeActivity一個頁面註冊即可。
  • 2、設定HomeActivity的啟動模式為singleTask。

當我們需要退出的時候只需要startActivity(this,HomeActivity,class), 再傳送一個退出廣播。上面程式碼首先會把棧中HomeActivity之上的所有Activity移除出棧,然後接到廣播finish自己。一切OK ! 沒有彈框,不用考慮機型Rom適配。不會有記憶體問題,就是那麼的優雅,簡單!

五、SingleTask改版式

和一些小夥交流之後,很多小夥伴說註冊廣播略顯麻煩,在樓下的小夥伴提出了一種更簡單的方式,思路也很簡單,

  • 1、設定MainActivity的載入模式為singleTask
  • 2、重寫MainActivity中的onNewIntent方法
  • 3、需要退出時在Intent中新增退出的tag

由於很多小夥伴對原始碼需求比較熱切,我們這裡就直接以程式碼的形式為大家講解這種方式

第一步設定MainActivity的載入模式為singleTask

 android:launchMode="singleTask"

第二步重寫onNewIntent()方法

    private static final String TAG_EXIT = "exit";

    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        if (intent != null) {
            boolean isExit = intent.getBooleanExtra(TAG_EXIT, false);
            if (isExit) {
                this.finish();
            }
        }
    }

第三步 退出

Intent intent = new Intent(this,MainActivity.class);
        intent.putExtra(MainActivity.TAG_EXIT, true);
        startActivity(intent);

六、懶人式

這種方式更加簡單,只需要如下兩步操作

  • 1、將MainActivity設定為singleTask
  • 2、將退出出口放置在MainActivity

我們可以看到很多應用都是雙擊兩次home鍵退出應用,就是基於這樣的方式來實現的,這裡在貼一下如何處理連續兩次點選退出的原始碼

private boolean mIsExit;
@Override
    /**
     * 雙擊返回鍵退出
     */
    public boolean onKeyDown(int keyCode, KeyEvent event) {

        if (keyCode == KeyEvent.KEYCODE_BACK) {
            if (mIsExit) {
                this.finish();

            } else {
                Toast.makeText(this, "再按一次退出", Toast.LENGTH_SHORT).show();
                mIsExit = true;
                new Handler().postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        mIsExit = false;
                    }
                }, 2000);
            }
            return true;
        }

        return super.onKeyDown(keyCode, event);
    }

相關文章