Android退出應用最優(yōu)雅的方式
我們先來看看幾種常見的退出方法(不優(yōu)雅的方式)
一、容器式
建立一個全局容器,把所有的Activity存儲起來,退出時循環(huán)遍歷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 activityStack = new ArrayList();
- 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 if (null != activityStack.get(i)) {
- activityStack.get(i).finish();
- }
- }
- activityStack.clear();
- }
- }
這種方法比較簡單, 但是可以看到activityStack持有這Activity的強引用,也就是說當某個Activity異常退出時,activityStack沒有即使釋放掉引用,就會導致內存問題,接下來我們看一種類似的方式,但是會稍微優(yōu)雅一點點
二、廣播式
通過在BaseActivity中注冊一個廣播,當退出時發(fā)送一個廣播,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優(yōu)雅式
什么是RS式呢?即Receiver+singleTask 。我們知道Activity有四種加載模式,而singleTask就是其中的一種,使用這個模式之后,當startActivity時,它先會在當前棧中查詢是否存在Activity的實例,如果存在,則將其至于棧頂,并將其之上的所有Activity移除棧。我們打開一個app,首先是一個splash頁面,然后會finish掉splash頁面。跳轉到主頁。然后會在主頁進行N次的跳轉,期間會產生數(shù)量不定的Activity,有的被銷毀,有的駐留在棧中,但是棧底永遠是我們的HomeActivity。這樣就讓問題變得簡單很多了。我們只需兩步操作即可優(yōu)雅的實現(xiàn)app的退出。
1、在HomeActivity注冊一個退出廣播,和第二個廣播式一樣,但是這里只需要在HomeActivity一個頁面注冊即可。
2、設置HomeActivity的啟動模式為singleTask。
當我們需要退出的時候只需要startActivity(this,HomeActivity,class), 再發(fā)送一個退出廣播。上面代碼首先會把棧中HomeActivity之上的所有Activity移除出棧,然后接到廣播finish自己。一切OK ! 沒有彈框,不用考慮機型Rom適配。不會有內存問題,就是那么的優(yōu)雅,簡單!
五、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鍵退出應用,就是基于這樣的方式來實現(xiàn)的,這里在貼一下如何處理連續(xù)兩次點擊退出的源碼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);
- }