手持ActivityLifecycleCallbacks,监听前后台状态如此舒服

2024-04-03 14:18

本文主要是介绍手持ActivityLifecycleCallbacks,监听前后台状态如此舒服,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

之前写过一小文,Android 程序打开即启动指定页面,比如密码检查页。,实现方式麻烦琐碎,后来Alex_Cin留言说可以考虑采用Application.ActivityLifecycleCallbacks来实现 监听程序处于前台还是后台 ,阅读学习后,大呼过瘾。很早就想把这个写成博客记录分享给各位朋友,但是因为个人原因迟迟未完成,现在补上。

关于监听程序处于前台还是后台

  • 弄一个BaseActivity,onResume等方法加1减1?这样不好,bug还是有的。
  • 获得所有程序的列表,判断当前程序是否至于前台?不好,而且新SDK有所限制。
  • 广播+服务,牛刀杀鸡。

所以,Application.ActivityLifecycleCallbacks登场,清脆利索,简单大气。

老规矩,先看图:

ges.gif

结构图

Paste_Image.png

菜已上,酒已喝。说说菜怎么做的吧。

一、Application.ActivityLifecycleCallbacks

Application通过ActivityLifecycleCallbacks使用接口提供了一套回调方法,用于让开发者对Activity的生命周期事件进行集中处理。 ActivityLifecycleCallbacks接口回调可以简化监测Activity的生命周期事件,在一个类中作统一处理。 ActivityLifecycleCallbacks使用要求API 14+ (Android 4.0+)。

既然是接口,那么来实现以下看看怎样呗

随便写一个类,比如名为TestAppCb,实现Application.ActivityLifecycleCallbacks接口

public class TestAppCb implements Application.ActivityLifecycleCallbacks {@Overridepublic void onActivityCreated(Activity activity, Bundle savedInstanceState) {}@Overridepublic void onActivityStarted(Activity activity) {}@Overridepublic void onActivityResumed(Activity activity) {}@Overridepublic void onActivityPaused(Activity activity) {}@Overridepublic void onActivityStopped(Activity activity) {}@Overridepublic void onActivitySaveInstanceState(Activity activity, Bundle outState) {}@Overridepublic void onActivityDestroyed(Activity activity) {}
}

我们发现了很多跟Activity很声明周期很相似的方法。作用不赘述,下面会附上一个名为ForegroundCallbacks的实现类,自行参考。

以往若需监测Activity的生命周期事件代码,你可能是这样做的,重写每一个Acivity的onResume(),然后作统计和处理,ActivityLifecycleCallbacks接口回调可以简化这一繁琐过程,在一个类中作统一处理。

二、具体使用

核心封装类,根据下面的代码的封装,结合自定义的Application,就可以轻松地实现监听程序处于前台还是后台的需求。

public class ForegroundCallbacks implements Application.ActivityLifecycleCallbacks {public static final long CHECK_DELAY = 500;public static final String TAG = ForegroundCallbacks.class.getName();public interface Listener {public void onBecameForeground();public void onBecameBackground();}private static ForegroundCallbacks instance;private boolean foreground = false, paused = true;private Handler handler = new Handler();private List<Listener> listeners = new CopyOnWriteArrayList<Listener>();private Runnable check;public static ForegroundCallbacks init(Application application){if (instance == null) {instance = new ForegroundCallbacks();application.registerActivityLifecycleCallbacks(instance);}return instance;}public static ForegroundCallbacks get(Application application){if (instance == null) {init(application);}return instance;}public static ForegroundCallbacks get(Context ctx){if (instance == null) {Context appCtx = ctx.getApplicationContext();if (appCtx instanceof Application) {init((Application)appCtx);}throw new IllegalStateException("Foreground is not initialised and " +"cannot obtain the Application object");}return instance;}public static ForegroundCallbacks get(){if (instance == null) {throw new IllegalStateException("Foreground is not initialised - invoke " +"at least once with parameterised init/get");}return instance;}public boolean isForeground(){return foreground;}public boolean isBackground(){return !foreground;}public void addListener(Listener listener){listeners.add(listener);}public void removeListener(Listener listener){listeners.remove(listener);}@Overridepublic void onActivityResumed(Activity activity) {paused = false;boolean wasBackground = !foreground;foreground = true;if (check != null)handler.removeCallbacks(check);if (wasBackground){L.d ("went foreground");for (Listener l : listeners) {try {l.onBecameForeground();} catch (Exception exc) {L.d ("Listener threw exception!:"+exc.toString());}}} else {L.d ("still foreground");}}@Overridepublic void onActivityPaused(Activity activity) {paused = true;if (check != null)handler.removeCallbacks(check);handler.postDelayed(check = new Runnable(){@Overridepublic void run() {if (foreground && paused) {foreground = false;L.d ("went background");for (Listener l : listeners) {try {l.onBecameBackground();} catch (Exception exc) {L.d ("Listener threw exception!:"+exc.toString());}}} else {L.d ("still foreground");}}}, CHECK_DELAY);}@Overridepublic void onActivityCreated(Activity activity, Bundle savedInstanceState) {}@Overridepublic void onActivityStarted(Activity activity) {}@Overridepublic void onActivityStopped(Activity activity) {}@Overridepublic void onActivitySaveInstanceState(Activity activity, Bundle outState) {}@Overridepublic void onActivityDestroyed(Activity activity) {}
}

.
.

MyApplication 结合Application.ActivityLifecycleCallbacks

监听如此轻松

public class MyApplication extends Application{@Overridepublic void onCreate() {super.onCreate();ForegroundCallbacks.init(this);ForegroundCallbacks.get().addListener(new ForegroundCallbacks.Listener() {@Overridepublic void onBecameForeground() {L.d("当前程序切换到前台");if(CacheUtils.getBoolean(getApplicationContext(), MyConst.GESTRUE_IS_LIVE)){L.d("已经开启手势锁");Intent intent = new Intent(getApplicationContext(), CheckGesPwdActivity.class);intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP|Intent.FLAG_ACTIVITY_NEW_TASK);startActivity(intent);}else{}}@Overridepublic void onBecameBackground() {L.d("当前程序切换到后台");}});}
}

如上,即可,手势库可以直接引用做个小demo。

整个过程实现起来都是相当轻松的,亮点是ForegroundCallbacks,里面的代码值得咀嚼和学习。

本文粗糙了点,哈哈哈哈,不过我为什么感觉我还是挺良心的,哈哈哈哈。

附上github链接GestureLockMaster

三、单个Activity单独使用?

就在一个页面监听,当然可以。

public class TestLifeAc extends AppCompatActivity implements ForegroundCallbacks.Listener {@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_city_list);// 注册监听ForegroundCallbacks.get(this).addListener(this);}@Overrideprotected void onDestroy() {super.onDestroy();// 移除监听ForegroundCallbacks.get(this).removeListener(this);}@Overridepublic void onBecameForeground(Activity activity) {// 切换为前台}@Overridepublic void onBecameBackground(Activity activity) {//切换为后台}
}

四,参考学习

Is my Android app currently foreground or background?

 

 

原文地址:https://www.jianshu.com/p/e7f64e6bc2cc

这篇关于手持ActivityLifecycleCallbacks,监听前后台状态如此舒服的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/873140

相关文章

Flutter监听当前页面可见与隐藏状态的代码详解

《Flutter监听当前页面可见与隐藏状态的代码详解》文章介绍了如何在Flutter中使用路由观察者来监听应用进入前台或后台状态以及页面的显示和隐藏,并通过代码示例讲解的非常详细,需要的朋友可以参考下... flutter 可以监听 app 进入前台还是后台状态,也可以监听当http://www.cppcn

spring @EventListener 事件与监听的示例详解

《spring@EventListener事件与监听的示例详解》本文介绍了自定义Spring事件和监听器的方法,包括如何发布事件、监听事件以及如何处理异步事件,通过示例代码和日志,展示了事件的顺序... 目录1、自定义Application Event2、自定义监听3、测试4、源代码5、其他5.1 顺序执行

MySQL 中的服务器配置和状态详解(MySQL Server Configuration and Status)

《MySQL中的服务器配置和状态详解(MySQLServerConfigurationandStatus)》MySQL服务器配置和状态设置包括服务器选项、系统变量和状态变量三个方面,可以通过... 目录mysql 之服务器配置和状态1 MySQL 架构和性能优化1.1 服务器配置和状态1.1.1 服务器选项

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

通过prometheus监控Tomcat运行状态的操作流程

《通过prometheus监控Tomcat运行状态的操作流程》文章介绍了如何安装和配置Tomcat,并使用Prometheus和TomcatExporter来监控Tomcat的运行状态,文章详细讲解了... 目录Tomcat安装配置以及prometheus监控Tomcat一. 安装并配置tomcat1、安装

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的

SpringBoot整合Canal+RabbitMQ监听数据变更详解

《SpringBoot整合Canal+RabbitMQ监听数据变更详解》在现代分布式系统中,实时获取数据库的变更信息是一个常见的需求,本文将介绍SpringBoot如何通过整合Canal和Rabbit... 目录需求步骤环境搭建整合SpringBoot与Canal实现客户端Canal整合RabbitMQSp

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu