判断当前应用是在前台还是后台

12/14/220 更新 每日一问 | 如何判断应用退出,或者到后台了?


1. 应用场景

接收推送的时候,当我们的APP 收到一条推送的时候,可能在应用内,也可能在应用外。
在应用内的时候,显示一个 Dialog 之类,在应用外的时候,弹一个notifyCation.

2. 如何判断是在显示还是没有显示

2.1 方法一

以前 Android 提供 API 是 getRunningTasks(),拿到所有正在运行的 Task,一个个进行判断。 如果自己的 package 在里面,就是在应用内。
但是每一次都需要 for - loop,比较操蛋,不做介绍

2.1 方法二

Android 4.0 之后 提供 API Application.ActivityLifecycleCallbacks,生命周期执行到哪些,就会回调哪些,

public interface ActivityLifecycleCallbacks {
        void onActivityCreated(Activity activity, Bundle savedInstanceState);
        void onActivityStarted(Activity activity);
        void onActivityResumed(Activity activity);
        void onActivityPaused(Activity activity);
        void onActivityStopped(Activity activity);
        void onActivitySaveInstanceState(Activity activity, Bundle outState);
        void onActivityDestroyed(Activity activity);
    }
public class AppStatusTracker implements Application.ActivityLifecycleCallbacks{
	//用来记录App状态信息
}

每次生命周期变动都会回调,收集起来就可以知道当前应用的状态

如果在应用外,所有的 Activity 都应该 onStop() 掉,或者 onDestroyed(),如果是在应用内,则只有一个 Activity 是 onResumed()。

根据Activity 之间跳转,的生命周期记录

A - -> onPaused()
B - -> onCreated()
B - -> onStarted()
B - -> onResumed()
A - -> onStoped()
A - -> onDestoryed()

   //界面要显示时调用此方法,onCreate -- onResume 之间不该写大量代码,
    // 防止初始化时间过长会导致黑屏,
    //一旦此方法回调,就可以确定当前应用是在前台执行
    @Override
    public void onActivityResumed(Activity activity) {
        L.e(activity.toString()+" onActivityResumed");
        //Activity 跳转过程中,onPause -> onResume 之间需要时间,这时候收到消息推送,这算前台?后台?
        //onPause() 可能为 false,也可能是要跳转到另外一个Activity,另外一个Activity 去掉用onResumed 方法
        
        isForground = true;
        timestamp = 0l;
        isScreenOff = false;
    }
    
    @Override
    public void onActivityStarted(Activity activity) {
        L.e(activity.toString()+" onActivityStarted");
        activeCount ++;
    }

    @Override
    public void onActivityStopped(Activity activity) {
        L.e(activity.toString()+" onActivityStopped");
        activeCount --;
        if (activeCount == 0){
            isForground = false;//在后台执行
            timestamp = System.currentTimeMillis();
        }
    }
    
  

3. 代码

public class AppStatusTracker implements Application.ActivityLifecycleCallbacks{

    private boolean isForground;
    private int activeCount;//计数器,所有的Activity都执行了stop方法,那么count=0,app在后台执行

    private static final long MAX_INTERVAL = 5 * 60* 1000;
    private long timestamp;//在后台呆的时间够长,再切换到前台的时间差,超过最大值显示图形解锁
    private boolean isScreenOff;//图形解锁的特殊需求,屏幕锁住之后再亮起需要图形解锁

    private static AppStatusTracker tracker;
    private Application application;
    private DaemonReceiver receiver;
    private int mAppStatus = ConstantValues.STATUS_FORCE_KILLED;

    private AppStatusTracker(Application application) {
        this.application = application;
        application.registerActivityLifecycleCallbacks(this);//必须注册才能接受到该方法的回调
    }

    public static void init(Application application) {
        tracker = new AppStatusTracker(application);
    }

    public static AppStatusTracker getInstance() {
        return tracker;
    }

    public void setAppStatus(int status) {
        this.mAppStatus = status;
        if (status == ConstantValues.STATUS_ONLINE){
            if (receiver == null){
                IntentFilter filter = new IntentFilter();
                filter.addAction(Intent.ACTION_SCREEN_OFF);
                receiver = new DaemonReceiver();
                application.registerReceiver(receiver,filter);
            }else if (receiver != null){
                application.unregisterReceiver(receiver);
                receiver = null;
            }
        }
    }

    public int getAppStatus() {
        return this.mAppStatus;
    }

    public boolean isForground() {
        return isForground;
    }

    private void onScreenOff(boolean isScreenOff) {
        this.isScreenOff = isScreenOff;
    }

    public boolean checkIfShowGesture(){
        if (mAppStatus == ConstantValues.STATUS_OFFLINE){
            if (isScreenOff){
                return false;
            }
            if (timestamp != 0l && System.currentTimeMillis() - timestamp > MAX_INTERVAL) {
                return true;
            }
        }
        return false;
    }

    @Override
    public void onActivityStopped(Activity activity) {
        L.e(activity.toString()+" onActivityStopped");
        activeCount --;
        if (activeCount == 0){
            isForground = false;//在后台执行
            timestamp = System.currentTimeMillis();
        }
    }

    @Override
    public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
        L.e(activity.toString()+" onActivityCreated");
    }

    @Override
    public void onActivityStarted(Activity activity) {
        L.e(activity.toString()+" onActivityStarted");
        activeCount ++;
    }

    //界面要显示时调用此方法,onCreate -- onResume 之间不该写大量代码,
    // 防止初始化时间过长会导致黑屏,
    //一旦此方法回调,就可以确定当前应用是在前台执行
    @Override
    public void onActivityResumed(Activity activity) {
        L.e(activity.toString()+" onActivityResumed");
        //Activity 跳转过程中,onPause -> onResume 需要时间,这过程中算前台?后台?
        isForground = true;
        timestamp = 0l;
        isScreenOff = false;
    }

    @Override
    public void onActivityPaused(Activity activity) {
        L.e(activity.toString()+" onActivityPaused");
    }

    @Override
    public void onActivityDestroyed(Activity activity) {
        L.e(activity.toString()+" onActivityDestroyed");
    }

    @Override
    public void onActivitySaveInstanceState(Activity activity, Bundle outState) {

    }

    private class DaemonReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
            L.d("onReceive:" + action);
            if (Intent.ACTION_SCREEN_OFF.equals(action)) {
                AppStatusTracker.getInstance().onScreenOff(true);
            }
        }
    }
}

使用:

public class CustomApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();
        AppStatusTracker.init(this);
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值