Android 经典笔记之二:如何退出应用程序

目录介绍:

  1. 1.如何退出应用程序
  2. 1.1 第一种方式:以任务栈形式退出程序
  3. 1.2 第二种方式:任务管理器方法【简称:进程式】
  4. 1.3 第三种方式:跳转页面后销毁栈堆【SingTask式】
  5. 1.4 第四种方式:容器式退出程序【目前用的就是这种 】
  6. 1.5 第五种方式:广播式退出程序
  7. 1.6 第六种方式:懒人式退出程序

第一种方式:以任务栈形式退出程序

  • 1.思路分析: 在Android的Activity中,当有多个Activity多次跳转后,点击返回或退出按钮会发现它会出现循环,会将刚刚打开的Activity,打开了多少次,一次次返回回去,当回到最开始的Activity后才会回到桌面 想要应用程序完全退出,可定义一个栈,利用单列模式管理Activity,写一个自定义Application类。 在每个Activity的onCreate()方法中调用自定义Application.getInstance().addActivity(this);方法,将该Activity添加到ExitApplication实例中,在要退出的地方调用自定义Application.getInstance().exit();方法,从而将整个应用程序完全退出。

  • 2.首先自定义Application类

public class App extends Application {
    private List<Activity> activityList = new LinkedList<>();
    private static App instance;

    public App() {

    }

    // 单例模式中获取唯一的ExitApplication实例
    public static App getInstance() {
        if (null == instance) {
            instance = new App();
        }
        return instance;
    }

    @Override
    public void onCreate() {
        super.onCreate();
    }

    // 将Activity添加到容器中
    public void addActivity(Activity activity) {
        activityList.add(activity);
    }

    // 当要退出Activity时,遍历所有Activity 并finish
    public void exit() {
        for (Activity activity : activityList) {
            activity.finish();
        }
        System.exit(0);
    }
}
  • 3.在父类BaseActivity中添加继承子类Activity到栈中【注意是所有Activity中必须添加】
//将该Activity添加到自定义Application实例中,
public class BaseActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 添加Activity到堆栈
        App.getInstance().addActivity(this);     //这句话必须添加
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 结束Activity&从栈中移除该Activity
        //App.getInstance().removeActivity(this);          //这句话建议不添加,因为有时返回上一个Activity不希望被销毁
    }
}

  • 4.在主页面执行退出程序方法
// 物理返回键,双击退出
private long exitTime = 0;
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_DOWN) {
        if ((System.currentTimeMillis() - exitTime) > 2000) {
            Toast.makeText(MainActivity.this, "再按一次退出程序", Toast.LENGTH_SHORT).show();
            exitTime = System.currentTimeMillis();
        } else {
            App.getInstance().exit();
        }
        return true;
    }
    return super.onKeyDown(keyCode, event);
}

第二种方式:任务管理器方法

  • 2.1 方法比较简单,但不太好使用
// 物理返回键,双击退出
private long exitTime = 0;
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_DOWN) {
        if ((System.currentTimeMillis() - exitTime) > 2000) {
            Toast.makeText(MainActivity.this, "再按一次退出程序", Toast.LENGTH_SHORT).show();
            exitTime = System.currentTimeMillis();
        } else {
            //系统会将,该包下的 ,所有进程,服务,全部杀掉,就可以杀干净了,要注意加上权限
            //<uses-permission android:name="android.permission.RESTART_PACKAGES"/>
            ActivityManager am = (ActivityManager)getSystemService (Context.ACTIVITY_SERVICE);
            am.restartPackage(getPackageName());
        }
        return true;
    }
    return super.onKeyDown(keyCode, event);
}

第三种方式:历史栈销毁所有Activity

  • 3.1 思路:通过Intent的Flags来控制堆栈去解决 Android的窗口类提供了历史栈,我们可以通过stack的原理来巧妙的实现,这里我们在A窗口打开B窗口时在Intent中直接加入标 志 Intent.FLAG_ACTIVITY_CLEAR_TOP,这样开启B时将会清除该进程空间的所有Activity。 android中,每打开一个Activity,便会在栈中加入一个Activity,当该Activity被摧毁后,栈中便移除了它,并且栈中的Activity是按照开打的先后顺序依次排排列的。 Android的窗口类提供了历史栈,我们可以通过stack的原理来巧妙的实现,这里我们在A窗口打开B窗口时在Intent中直接加入标志 Intent.FLAG_ACTIVITY_CLEAR_TOP,这样开启B时将会清除该进程空间的所有Activity。
Intent intent = new Intent(this,MainActivity.class); 
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); 
startActivity(intent);

或者,下面这种方式也可以可以的,简单快捷

  • 1、设置MainActivity的加载模式为singleTask
  • 2、重写MainActivity中的onNewIntent方法
  • 3、需要退出时在Intent中添加退出的tag
// 物理返回键,双击退出
private long exitTime = 0;
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_DOWN) {
        if ((System.currentTimeMillis() - exitTime) > 2000) {
            Toast.makeText(MainActivity.this, "再按一次退出程序", Toast.LENGTH_SHORT).show();
            exitTime = System.currentTimeMillis();
        } else {
            exit();
        }
        return true;
    }
    return super.onKeyDown(keyCode, event);
}

/**退出程序**/
protected void exit() {
    // 退出程序方法有多种
    // 这里使用clear + new task的方式清空整个任务栈,只保留新打开的Main页面
    // 然后Main页面接收到退出的标志位exit=true,finish自己,这样就关闭了全部页面
    Intent intent = new Intent(this, MainActivity.class);
    intent.putExtra("exit", true);
    startActivity(intent);
}

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

第四种:容器式退出程序【目前用的就是这种】

  • 建立一个全局容器,把所有的Activity存储起来,退出时循环遍历finish所有Activity
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&从栈中移除该Activity
        //AtyContainer.getInstance().removeActivity(this);
    }
}

class AtyContainer {

    public AtyContainer() {
    }

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

    public static AtyContainer getInstance() {
        return instance;
    }

    /**
     * 添加Activity到堆栈
     */
    public void addActivity(Activity activity) {
        if (activityStack == null) {
            activityStack = new ArrayList<Activity>();
        }
        activityStack.add(activity);
    }

    /**
     * 移除指定的Activity
     */
    public void removeActivity(Activity activity) {
        if (activity != null) {
            activityStack.remove(activity);
            activity = null;
        }
    }
    /**
    * 结束所有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();
        }
    }

}

第六种方式:懒人式退出程序

  • 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);
    }

后续: 平时喜欢写写文章,笔记。别人建议我把笔记,以前写的东西整理,然后写成博客,所以我会陆续整理文章,只发自己写的东西,敬请期待:

  •       知乎:https://www.zhihu.com/people/yang-chong-69-24/pins/posts
    
  •       领英:https://www.linkedin.com/in/chong-yang-049216146/
    
  •       简书:http://www.jianshu.com/u/b7b2c6ed9284
    
  •       csdn:http://my.csdn.net/m0_37700275
    
  •       网易博客:http://yangchong211.blog.163.com/
    
  •       新浪博客:http://blog.sina.com.cn/786041010yc
    
  •       github:https://github.com/yangchong211
    
  •       喜马拉雅听书:http://www.ximalaya.com/zhubo/71989305/
    
  •       脉脉:yc930211
    
  •       开源中国:https://my.oschina.net/zbj1618/blog
    

转载于:https://my.oschina.net/zbj1618/blog/1510669

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值