说到App的前后台监控,有什么使用场景呢?例如:

IM模块,收到消息时,需要判断当前App是否在前台,如果在前台则震动一下提醒用户,如果在后台则发送一条通知提醒用户。

收到Push推送,需要判断App是否存活,如果存活则直接跳转到目标界面,如果不存活则先启动App,再跳转到目标页面。

有没有简便的Api

iOS上的AppDelegate

applicationDidEnterBackground方法回调提醒App处于后台。

applicationWillEnterForeground方法回调提醒App处于前台。

Android的Application

很遗憾,没有!

既然没有,有没有别的法子?

解决方案

其实获取方法,有好几种:

通过ActivityManager.getRunningTasks()获取当前运行的task列表,需要遍历,而且需要申请android.permission.GET_TASKS权限,而且5.0后被废弃,5.0以上版本失效。

5.0后,ActivityManager提供了getRunningAppProcesses()方法获取正在运行的应用程序,同样需要遍历,但不需要申请权限,但据说8.0后失效了。

Application注册ActivityLifecycleCallbacks,获取App上的所有Activity生命周期改变的回调,对onActivityStarted和onActivityStopped做计数即可,如果计数为0代表为后台,大于0则为前台,需要Api14以上,就是4.0以上可使用。一般现在兼容都在4.4,所以基本没有兼容性问题,并且不需要申请权限。

方案选择

基于申请权限和不需要遍历的性能消耗,我选择了方案3。

新建一个单例类AppMonitor,在Application中初始化,并注册ActivityLifecycleCallbacks回调,我们再对生命周期回调进行计数,并且提供我们自定义的回调即可。

提供initialize方法初始化AppMonitor。

提供Callback接口

提供onAppForeground回调,表示App切换到前台。

提供onAppBackground回调,表示App切换到后台。

提供onAppUIDestroyed回调,表示App所有界面都销毁了,可以认为就是App被“杀死”了。

public class AppMonitor {

/**

* 注册了的监听器

*/

private List mCallbacks;

/**

* 是否初始化了

*/

private boolean isInited;

/**

* 活跃Activity的数量

*/

private int mActiveActivityCount = 0;

/**

* 存活的Activity数量

*/

private int mAliveActivityCount = 0;

public static AppMonitor get() {

return SingleHolder.INSTANCE;

}

private static final class SingleHolder {

private static final AppMonitor INSTANCE = new AppMonitor();

}

public void initialize(Context context) {

if (isInited) {

return;

}

mCallbacks = new CopyOnWriteArrayList<>();

registerLifecycle(context);

isInited = true;

}

/**

* 注册生命周期

*/

private void registerLifecycle(Context context) {

Application application = (Application) context.getApplicationContext();

application.registerActivityLifecycleCallbacks(new Application.ActivityLifecycleCallbacks() {

@Override

public void onActivityCreated(Activity activity, Bundle savedInstanceState) {

mAliveActivityCount++;

}

@Override

public void onActivityStarted(Activity activity) {

mActiveActivityCount++;

notifyChange();

}

@Override

public void onActivityResumed(Activity activity) {

}

@Override

public void onActivityPaused(Activity activity) {

}

@Override

public void onActivityStopped(Activity activity) {

mActiveActivityCount--;

notifyChange();

}

@Override

public void onActivitySaveInstanceState(Activity activity, Bundle outState) {

}

@Override

public void onActivityDestroyed(Activity activity) {

mAliveActivityCount--;

notifyAppAliveChange();

}

});

}

/**

* 判断App是否活着

*/

public boolean isAppAlive() {

return mAliveActivityCount > 0;

}

/**

* 判断App是否在前台

*/

public boolean isAppForeground() {

return mActiveActivityCount > 0;

}

/**

* 判断App是否在后台

*/

public boolean isAppBackground() {

return mActiveActivityCount <= 0;

}

/**

* 通知监听者

*/

private void notifyChange() {

if (mActiveActivityCount > 0) {

for (Callback callback : mCallbacks) {

callback.onAppForeground();

}

} else {

for (Callback callback : mCallbacks) {

callback.onAppBackground();

}

}

}

/**

* 通知监听者界面销毁

*/

private void notifyAppAliveChange() {

if (mAliveActivityCount == 0) {

for (Callback callback : mCallbacks) {

callback.onAppUIDestroyed();

}

}

}

public interface Callback {

/**

* 当App切换到前台时回调

*/

void onAppForeground();

/**

* App切换到后台时回调

*/

void onAppBackground();

/**

* App所有界面都销毁了

*/

void onAppUIDestroyed();

}

public static class CallbackAdapter implements Callback {

@Override

public void onAppForeground() {

}

@Override

public void onAppBackground() {

}

@Override

public void onAppUIDestroyed() {

}

}

/**

* 注册回调

*/

public void register(Callback callback) {

if (mCallbacks.contains(callback)) {

return;

}

mCallbacks.add(callback);

}

/**

* 注销回调

*/

public void unRegister(Callback callback) {

if (!mCallbacks.contains(callback)) {

return;

}

mListener.remove(callback);

}

}

使用

初始化

public class MyApplication extends Application {

@Override

public void onCreate() {

super.onCreate();

//前后台监听初始化

AppMonitor.get().initialize(this);

}

}

注册监听和注销监听

AppMonitor.Callback callback = new AppMonitor.Callback() {

@Override

public void onAppForeground() {

//App切换到前台

}

@Override

public void onAppBackground() {

//App切换到后台

}

@Override

public void onAppUIDestroyed() {

//App被杀死了

}

};

//注册回调

AppMonitor.get().register(callback);

//注销回调

AppMonitor.get().unRegister(callback);

判断App是否在前台

boolean isAppForeground = AppMonitor.get().isAppForeground();

判断App是否在后台

boolean isAppBackground = AppMonitor.get().isAppBackground();

判断App是否存活(所有页面都销毁了)

boolean isAlive = AppMonitor.get().isAppAlive();

总结

虽然Android中没有简洁明了的Api,获取前后台状态和回调,但是好在我们可以自己动手丰衣足食,满足需求。

android监控app被杀死,Android App前后台监控相关推荐

  1. android 通过广播唤醒被杀死的app

    今天,简单讲讲如何唤醒被杀死的app. 这个唤醒app主要通过广播来唤醒. 1. 静态广播唤醒 广播的exported属性和enabled属性 exported默认为true表示这个广播可以接收来自其 ...

  2. Android 系统(274)---Anroid5.0以上进程保活方案(亲测可自行调起被杀死的app)

    Anroid5.0以上进程保活方案(亲测可自行调起被杀死的app) 实验了几种最后选择了2中组合 第一种 开启一像素activity保活,如果2个activity同时被杀死则此方案不可选. 第二种 发 ...

  3. Android进程保活(如何尽可能避免APP被杀死)

    #.Android进程的优先级 Android系统中进程有不同的优先级,在系统需要优化和回收资源时,会先杀死优先级低的进程.所以要保活一个进程,就要想办法尽可能去提升它的优先级. ##1.前台进程 - ...

  4. Android P 性能优化:创建APP进程白名单,杀死白名单之外的进程

    一.前言 最近开发过程中,因系统内存相对比较紧张,在启动某些CPU.内存占用比较大的APP的时候,就需要清理一下,把能杀的都杀掉,给前台进程腾出系统资源. 缺陷:对于persist进程,以及一些流氓A ...

  5. 【Android】一个APP检测另一个APP的Service被杀死时自动重启服务

    例如:appA要检测启动appB中的service ##1.修改B中Service启动时的FLAG @Overridepublic int onStartCommand(Intent intent, ...

  6. android下载后的app自动安装,Android 7.0 下载APK后自动安装

    随着Android版本越来越高,Android对隐私的保护力度也越来越大.这些隐私权限的更改在为用户带来更加安全的操作系统的同时也为开发者带来了一些新的任务.如何让你的APP能够适应这些改变而不是崩溃 ...

  7. Android 应用开发(28)----APP功耗测试方法

    APP功耗测试方法 业界比较难测试的一个测试,比如新旧版本对比不一定能说明问题,所以耗电量测试只能定量测试,作为辅助数据帮助我们测试.功耗测试可以基于硬件测试方法(第三方精密仪器)和基于软件测评方法. ...

  8. Android系统性能优化(56)---APP性能优化

    Android客户端性能优化(魅族资深工程师毫无保留奉献) 转载学习:http://blog.tingyun.com/web/article/detail/155?from=groupmessage& ...

  9. Android 性能优化---(8)APP启动时间优化指南

    本文可以帮助你优化应用的启动时间:首先描述应用启动过程的内部机制:然后讨论如何分析启动性能:最后,列举了一些常见的影响启动时间的问题,并就如何解决这些问题给出一些提示. 第 1 部分:启动过程内部机制 ...

最新文章

  1. 腾讯优图吴永坚:迈向深度学习,我们面临模型训练与推荐的双重考验
  2. AI促进药物发现:未来是多细胞研究
  3. 【C 语言】二级指针 内存模型图 ( 指针数组 | 二维数组 | 自定义二级指针内存 )
  4. JAVA test代码运行
  5. Linux 基础学习:文件权限与种类
  6. 谈操作系统的碎片化和融合
  7. Day05:装饰器,三元表达式,函数的递归,匿名/内置函数,迭代器,模块,开发目录...
  8. 【caffe】ubuntu配置matlab接口----matcaffe
  9. silverlight 3 blend3最新版本 破解方法
  10. 2016SEM行业现状
  11. go 服务器压力测试,Go的单元测试与压力测试
  12. 思维-思维方法:思维方法
  13. Rsync: @ERROR: Auth Failed On Module XXX错误原因及解决办法
  14. 闲聊2022卡塔尔世界杯
  15. Ja进av阶书籍推荐
  16. 使用 Amazon Amplify快速创建简单的 Android 应用程序
  17. 阿里云天池大赛赛题(机器学习)——阿里云安全恶意程序检测(完整代码)
  18. 报告总结:无线通信中的数学问题
  19. 51单片机驱动DS18B20温度传感器测量温度
  20. 图解Linux命令之--mkfs命令

热门文章

  1. 计算机病毒可通过光盘传播吗,光盘能传播病毒吗
  2. 表空间相关命令及查询SQL
  3. 如何用python爬股票数据_python爬虫股票数据,如何用python 爬虫抓取金融数据
  4. 阿里云的认证是不是最好的?考起来难不难?
  5. 双11购物狂欢已经开始,店宝宝:火热的直播电商正待加码
  6. 使用微搭低代码集成腾讯地图
  7. 工业机器人码垛教学实施_《工业机器人码垛工作站安装与调试》教学设计文本.doc...
  8. 嵌入式Linux开发
  9. python多线程中join()的理解
  10. 1.6 电源树中电流的计算方法(硬件基础系列)