在开发过程中,我们常常需要一个退出功能,来退出该应用的所有Activity。下面,我们列举一些退出应用的几种方式。以下用的源码点击查看源码地址
欢迎star,欢迎fork
- 利用ActivityContainer来管理所有的Activity的引用
- 使用广播通知BaseActivity结束
- 直接杀死进程
- 采用SingleTask的特点,结束应用
- 第四种方式的更优雅实现
- 双击返回键退出
方式一:利用ActivityContainer管理所有的Activity引用
该方式是建立一个通用的Activity的父类 BaseActivity
以及一个ActivityContainer
类,ActivityContainer
是一个集合类来保存所有Activity的引用,关键的代码如下:
public class BaseActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityContainer.getInstance().addActivity(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
ActivityContainer.getInstance().removeActivity(this);
}
}
class ActivityContainer {
private ActivityContainer(){
}
private static ActivityContainer instance = new ActivityContainer();
private static List<AppCompatActivity> activityStack = new LinkedList<>();
public static ActivityContainer getInstance(){
return instance;
}
public void addActivity(AppCompatActivity aty) {
activityStack.add(aty);
}
public void removeActivity(AppCompatActivity 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();
}
}
然后真正的业务需求的Activity分别继承自BaseActivity,在需要退出应用的时候,执行ActivityContainer.getInstance().finishAllActivity();
方式二:使用广播通知BaseActivity结束
在原有的BaseActivity的基础上,创建一个广播接收器,当收到广播通知的时候,执行BaseActivity.this.finish();
BaseActivity的代码如下:
public class BaseActivity extends AppCompatActivity {
protected static final String EXITACTION = "action.exit";
private ExitReceiver mExitReceiver = new ExitReceiver();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(EXITACTION);
registerReceiver(mExitReceiver, intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(mExitReceiver);
}
class ExitReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
BaseActivity.this.finish();
}
}
}
方式三:直接杀死进程
该方法是通过系统提供的杀死进程的方法来杀死当前Activity所在的进程,但由于各种定制ROM对底层修改的太多,导致很多时候该方法并不能奏效
方式一 android.os.Process.killProcess(android.os.Process.myPid());
方式二 System.exit(0);
方式三 ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
manager.killBackgroundProcesses(getPackageName());
方式四:采用SingleTask的特点,结束应用
我们的应用一般的Activity打开顺序是最开始的splash页面,然后调转到我们的主Activity,同时finish掉splash页面,然后在主Activity上面进行各种Activity的跳转,这些跳转的Activity,有的finish掉,有的在任务栈中。如果我们设定主Activity的启动模式是singleTask,然后从该Activity启动的Activity的启动模式是默认模式,则这些Acitivty都会和主Activity在一个任务栈中,并且主Activity会在栈底。这时,当我们退出的时候,启动主Activity,并且给主Activity发送一个退出Acitivity的广播,则因为singleTask的特点,会清空其上所有的Activity上面的所有Activity,广播会通知主Activity结束自己,则就完整的退出了所有的Activity。
具体的步骤如下:
- 在MainActivity中注册一个广播,该广播内容是
MainActivity.this.finish()
- 当在选项页面,点击退出按钮时,启动MainActivity,会因为singleTask的特点,清空掉其他的Activity,在启动的时候,send一个广播,给MainActivity,让其结束。
- 退出全部应用
方式五:第四种方式的更优雅实现
第四种方式的思路是正确的,但是最后通过广播来通知MainActivity退出的形式,确实不是很好。对于Activity的实现,有更好的方式,就是借助onNewIntent
的方法。
该方法是在Activity重新从后台回到前台的时候会调用,完整的方法调用是 onNewIntent-onRestart-onResume
。我们可以在onNewIntent
中实现MainActivity的退出。
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
Log.e("main", "onNewIntent调用");
if (intent != null) {
boolean isExit = intent.getBooleanExtra(ACTION, false);
if (isExit) {
this.finish();
}
}
}
当在选项页面,点击退出按钮时,启动MainActivity时,Intent携带一个退出的参数,当检测到该参数的时候,退出应用。
方式六:双击返回键退出
双击退出键是比较简答有效的一种退出方式,比较适用于退出的界面是MainActivity,利用MainActivity的singleTask的特性,但回到前台的时,清空其上的其他Activity。这时,在MainActivity上监听keyDown事件,双击退出。
代码如下:
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (keyCode == KeyEvent.KEYCODE_BACK) {
if (isExit) {
this.finish();
}else {
Toast.makeText(this, "再按一次退出", Toast.LENGTH_SHORT).show();
isExit = true;
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
isExit = false;
}
},2000);
return true;
}
}
return super.onKeyDown(keyCode, event);
}
在这里,是由一个handler来实现一个计时器的功能,保证2s之内,不过不在返回,则恢复状态。这个计时器有多种实现方式,比如计时器,如下:
new Timer().schedule(new TimerTask() {
@Override
public void run() {
isExit = false;
}
},2000);
又或者简单的计算时间差,该方式实在太多简单,就不多说了。