手持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

相关文章

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

C#监听txt文档获取新数据方式

《C#监听txt文档获取新数据方式》文章介绍通过监听txt文件获取最新数据,并实现开机自启动、禁用窗口关闭按钮、阻止Ctrl+C中断及防止程序退出等功能,代码整合于主函数中,供参考学习... 目录前言一、监听txt文档增加数据二、其他功能1. 设置开机自启动2. 禁止控制台窗口关闭按钮3. 阻止Ctrl +

IIS 7.0 及更高版本中的 FTP 状态代码

《IIS7.0及更高版本中的FTP状态代码》本文介绍IIS7.0中的FTP状态代码,方便大家在使用iis中发现ftp的问题... 简介尝试使用 FTP 访问运行 Internet Information Services (IIS) 7.0 或更高版本的服务器上的内容时,IIS 将返回指示响应状态的数字代

Kotlin Compose Button 实现长按监听并实现动画效果(完整代码)

《KotlinComposeButton实现长按监听并实现动画效果(完整代码)》想要实现长按按钮开始录音,松开发送的功能,因此为了实现这些功能就需要自己写一个Button来解决问题,下面小编给大... 目录Button 实现原理1. Surface 的作用(关键)2. InteractionSource3.

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

基于Java实现回调监听工具类

《基于Java实现回调监听工具类》这篇文章主要为大家详细介绍了如何基于Java实现一个回调监听工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录监听接口类 Listenable实际用法打印结果首先,会用到 函数式接口 Consumer, 通过这个可以解耦回调方法,下面先写一个

SpringSecurity JWT基于令牌的无状态认证实现

《SpringSecurityJWT基于令牌的无状态认证实现》SpringSecurity中实现基于JWT的无状态认证是一种常见的做法,本文就来介绍一下SpringSecurityJWT基于令牌的无... 目录引言一、JWT基本原理与结构二、Spring Security JWT依赖配置三、JWT令牌生成与

关于WebSocket协议状态码解析

《关于WebSocket协议状态码解析》:本文主要介绍关于WebSocket协议状态码的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录WebSocket协议状态码解析1. 引言2. WebSocket协议状态码概述3. WebSocket协议状态码详解3