1、容器式java
創建一個全局容器,把全部的Activity存儲起來,退出時循環遍歷finish全部Activityandroid
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 ListactivityStack = 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 < size; i++) { if (null != activityStack.get(i)) { activityStack.get(i).finish(); } } activityStack.clear(); } }
這種方法比較簡單, 可是能夠看到activityStack持有這Activity的強引用,也就是說當某個Activity異常退出時,activityStack沒有即便釋放掉引用,就會致使內存問題,接下來咱們看一種相似的方式,可是會稍微優雅一點點app
2、廣播式可利用eventbuside
經過在BaseActivity中註冊一個廣播,當退出時發送一個廣播,finish退出post
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(); } } }
3、進程式this
經過直接殺死當前應用的進程來結束應用,簡單粗暴,並且有(wu)效!spa
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層改動太多,使用這種方式需慎重) 。code
4、 RS優雅式blog
什麼是RS式呢?即Receiver+singleTask 。咱們知道Activity有四種加載模式,而singleTask就是其中的一種,使用這個模式以後,當startActivity時,它先會在當前棧 中查詢是否存在Activity的實例,若是存在,則將其至於棧頂,並將其之上的全部Activity移除棧。咱們打開一個app,首先是一個 splash頁面,而後會finish掉splash頁面。跳轉到主頁。而後會在主頁進行N次的跳轉,期間會產生數量不定的Activity,有的被銷 毀,有的駐留在棧中,可是棧底永遠是咱們的HomeActivity。這樣就讓問題變得簡單不少了。咱們只需兩步操做便可優雅的實現app的退出。進程
一、在HomeActivity註冊一個退出廣播,和第二個廣播式同樣,可是這裏只須要在HomeActivity一個頁面註冊便可。
二、設置HomeActivity的啓動模式爲singleTask。
當咱們須要退出的時候只須要startActivity(this,HomeActivity,class), 再發送一個退出廣播。上面代碼首先會把棧中HomeActivity之上的全部Activity移除出棧,而後接到廣播finish本身。一切OK ! 沒有彈框,不用考慮機型Rom適配。不會有內存問題,就是那麼的優雅,簡單!
5、SingleTask改版式
和一些小夥交流以後,不少小夥伴說註冊廣播略顯麻煩,在樓下的小夥伴提出了一種更簡單的方式,思路也很簡單,
一、設置MainActivity的加載模式爲singleTask
二、重寫MainActivity中的onNewIntent方法
三、須要退出時在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);
6、懶人式
這種方式更加簡單,只須要以下兩步操做
一、將MainActivity設置爲singleTask
二、將退出出口放置在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); }