Android 架构组件之 Lifecycle

2024-02-25 16:58

本文主要是介绍Android 架构组件之 Lifecycle,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 1. 为什么要使用 Lifecycle?
      • 1.1 在 Lifecycle 出现之前,实现 Activity 与 Presenter 的绑定
      • 1.2 使用 Lifecycle 完成 Activity 与 Presenter 的绑定
    • 2. 分析 Lifecycle 的组成及实现原理
      • 2.1 Lifecycle 在 Activity 中的使用
      • 2.2 Lifecycle 在 Fragment 中的使用
    • 3. 总结一下 Lifecycle 的使用
    • 参考链接

Lifecycle 是 Google 给出的架构组件之一,它能够感知具有生命周期的组件(如 Activity 和 Fragment)的生命周期变化。本文将从以下几个方面对 Lifecycle 进行介绍:

  • 为什么要使用 Lifecycle?
  • 分析 Lifecycle 的组成及实现原理;
  • 总结一下 Lifecycle 的使用

1. 为什么要使用 Lifecycle?

Lifecycle 能够感知具有生命周期的组件(如 Activity 和 Fragment)的生命周期变化,
使用 Lifecycle 可以避免编写模板代码,同时生成组织性更好,更便于维护的代码。这是 Lifecycle 的特点,下面将从实际开发中的一个案例,来介绍为什么要使用 Lifecycle。

1.1 在 Lifecycle 出现之前,实现 Activity 与 Presenter 的绑定

我们经常见到在 Activity 和 Fragment 的生命周期中写一些业务相关的代码,这些代码通常组织性和可读性不高,维护起来也非常不方便。

这时候我们通常会用到 MVP 模式,将这些代码从相应的方法中剥离。比如我们会定义一个接口 IPresenter,在接口中定义一些生命周期相关的方法。

interface IPresenter {fun onCreate(savedInstanceState: Bundle?)fun onStart()fun onPause()fun onResume()fun onStop()fun onDestroy()
}

然后需要再定义一个类 MainPresenter,来实现 IPresenter 接口。

class MainPresenter :IPresenter{override fun onCreate() {}override fun onStart() {}override fun onPause() {}override fun onResume() {}override fun onStop() {}override fun onDestroy() {}
}

最后要在我们的 MainActivity 中的每个生命周期的回调方法中都要调用 IPresenter 中对应的方法。

class MainActivity : AppCompatActivity() {val presenter = MainPresenter()override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)presenter.onCreate()}override fun onStart() {super.onStart()presenter.onStart()}override fun onResume() {super.onResume()presenter.onResume()}override fun onPause() {super.onPause()presenter.onPause()}override fun onStop() {super.onStop()presenter.onStop()}override fun onDestroy() {super.onDestroy()presenter.onDestroy()}
}

当然这只是一种最简单的方式,实现将 Activity 与 Presenter 绑定,当然我们可以把绑定的过程通过反射来实现,这样就避免了重复的绑定。

那有没有更优雅的方式呢,答案是 Lifecycle。我们来看一下,使用 Lifecycle 该如何实现将 Activity 与 Presenter 的绑定呢?

1.2 使用 Lifecycle 完成 Activity 与 Presenter 的绑定

首先让 MainPresenter 实现 DefaultLifecycleObserver 接口,这里可以根据需要去实现生命周期的方法。

class MainPresenter : DefaultLifecycleObserver {override fun onCreate(owner: LifecycleOwner) {super.onCreate(owner)Logger.d("onCreate...")}override fun onStop(owner: LifecycleOwner) {super.onStop(owner)Logger.d("onStop...")}
}

然后在 MainActivity 中添加 MainPresenter 的实例,至此就完成了 Activity 与 Presenter 的绑定。

class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.layout_lifecycle)val presenter = MainPresenter()lifecycle.addObserver(presenter)}
}

可以认为 Google 将我们以前通过 MVP 框架实现的 V 与 P 的绑定过程 ,放到了Android 的系统框架里,并且功能丰富。

2. 分析 Lifecycle 的组成及实现原理

分析框架的组成和实现原理,我习惯从类图开始,在绘制类图的过程中,会了解框架大概包含哪些组成部分,有哪些比较重要的成员变量和方法。

2.1 Lifecycle 在 Activity 中的使用

我们以上面的示例 MainActivity,看一下 MainActivity、LifecycleOwner、Lifecycle 和 LifecycleObserver 之间的类图。在类图中对 LifecycleOwner、Lifecycle 和 LifecycleObserver 进行了颜色的区分,方便理解。

看过上面的类图,我们总结一下:

  • Lifecycle 是一个抽象类,它有两个枚举类型的内部类 Event和 State,分别对应着事件类型和组件的状态;
  • LifecycleOwner 是一个接口,它有一个 getLifecycle()方法要子类去实现。
  • 我们的 MainActivity 继承自 FragmentActivity,而 FragmentActivity 是 LifecycleOwner 的实现类。在 MainActivity 中 getLifecycle()方法返回的是 Lifecycle 的子类 LifecycleRegistry实例。
  • 在 LifecycleRegistry 类中的静态内部类 ObserverWithState 持有LifecycleEventObserver 对象的实例,LifecycleEventObserver 是 LifecycleObserver 的子类。

至此,MainActivity、LifecycleOwner、Lifecycle 和 LifecycleObserver 就都关联了起来。

**但是这几个类之间通过怎样的调用关系,实现了对 Activity 生命周期的监听呢?**再来看一张时序图。

看过上面的时序图,我们可以总结一下:

  • 在 MainActivity 中调用 getLifecycle()方法得到 LifecycleRegistry 对象,LifecycleRegistry 对象的初始化在 FragmentActivity 中完成。调用 addObserver()方法将 LifecycleObserver 的对象作为参数传递进去,这样就完成了一个完整的注册监听过程。

  • 在 Activity 生命周期变化时,在每个生命周期回调函数中调用 LifecycleRegistry的 handleLifecycleEvent()方法,在 handleLifecycleEvent()方法的层层调用中,最后通过 observer.dispatchEvent(lifecycleOwner, event) 将生命周期的变化传递到 LifecycleObserver 的实现类MainPresenter 中。

这个过程,参考 LifecycleRegistry 中的 handleLifecycleEvent()方法的调用关系。

public void handleLifecycleEvent(@NonNull Lifecycle.Event event) {State next = getStateAfter(event);moveToState(next);}private void moveToState(State next) {if (mState == next) {return;}mState = next;if (mHandlingEvent || mAddingObserverCounter != 0) {mNewEventOccurred = true;// we will figure out what to do on upper level.return;}mHandlingEvent = true;sync();mHandlingEvent = false;}private void sync() {LifecycleOwner lifecycleOwner = mLifecycleOwner.get();if (lifecycleOwner == null) {throw new IllegalStateException("LifecycleOwner of this LifecycleRegistry is already"+ "garbage collected. It is too late to change lifecycle state.");}while (!isSynced()) {mNewEventOccurred = false;// no need to check eldest for nullability, because isSynced does it for us.if (mState.compareTo(mObserverMap.eldest().getValue().mState) < 0) {backwardPass(lifecycleOwner);}Entry<LifecycleObserver, ObserverWithState> newest = mObserverMap.newest();if (!mNewEventOccurred && newest != null&& mState.compareTo(newest.getValue().mState) > 0) {forwardPass(lifecycleOwner);}}mNewEventOccurred = false;}private void backwardPass(LifecycleOwner lifecycleOwner) {Iterator<Entry<LifecycleObserver, ObserverWithState>> descendingIterator =mObserverMap.descendingIterator();while (descendingIterator.hasNext() && !mNewEventOccurred) {Entry<LifecycleObserver, ObserverWithState> entry = descendingIterator.next();ObserverWithState observer = entry.getValue();while ((observer.mState.compareTo(mState) > 0 && !mNewEventOccurred&& mObserverMap.contains(entry.getKey()))) {Event event = downEvent(observer.mState);pushParentState(getStateAfter(event));observer.dispatchEvent(lifecycleOwner, event);popParentState();}}}

2.2 Lifecycle 在 Fragment 中的使用

了解了在 Activity 中的实现过程,下面来看这个过程在 Fragment 中是怎样的。

看过类图,我们总结一下:

  • Fragment 继承了 LifecycleOwner 接口,getLifecycle()方法返回的是 Lifecycle 的子类 LifecycleRegistry实例。

  • 同样的,在 LifecycleRegistry 类中的静态内部类 ObserverWithState 持有LifecycleEventObserver 对象,LifecycleEventObserver 是 LifecycleObserver 的子类。

**那 LifecyleObserver 的实现类又是怎样对 Fragment 的生命周期进行监听的呢?**给出时序图:

总结一下:

  • 可以看到和对 Activity 生命周期监听过程是类似的,在 Fragment 中调用 getLifecycle()方法得到 LifecycleRegistry 对象,调用 addObserver()方法将实现了 LifecycleObserver 的对象作为参数传递进去,这样就完成了一个完整的注册监听过程。

  • 在 Fragment 生命周期变化时,在每个生命周期回调函数中调用 LifecycleRegistry的 handleLifecycleEvent()方法。在 handleLifecycleEvent()方法的层层调用中,最后通过 observer.dispatchEvent(lifecycleOwner, event) 将生命周期的变化传递到 LifecycleObserver 的实现类中。区别于在 Activity中的过程,Fragment 中生命周期的方法会先调用 performXXX()方法。

3. 总结一下 Lifecycle 的使用

介绍完 Lifecycle 的原理,最后来总结一下 Lifecyle 的使用:

1. 首先导入 Lifecycle 的依赖,Lifecycle 有很多依赖包,可以根据需要进行导入。

2.创建 LifecycleObserver 的实现类。
从上面的类图关系中可以看到,跟 LifecycleObserver 有关的接口有4个,分别是 LifecycleObserver、FullLifecycleObserver、DefaultLifecycleObserver 和LifecycleEventObserver。其中 FullLifecycleObserver 是一个包级别的接口,我们没有办法直接使用。所以实现 LifecycleObserver 就有三种方式。

  • 方式一,实现 DefaultLifecycleObserver 接口,然后重现接口中与生命周期有关的方法;
class MainPresenter : DefaultLifecycleObserver{override fun onCreate(owner: LifecycleOwner) {super.onCreate(owner)Logger.d("onCreate...")}override fun onStop(owner: LifecycleOwner) {super.onStop(owner)Logger.d("onStop...")}
}
  • 方式二,实现 LifecycleObserver 接口,然后通过注解的方式来监听生命周期的变化;
class MainPresenter : LifecycleObserver {@OnLifecycleEvent(Lifecycle.Event.ON_CREATE)fun onCreate() {Logger.d("onCreate...")}@OnLifecycleEvent(Lifecycle.Event.ON_START)fun onStart() {Logger.d("onStart...")}@OnLifecycleEvent(Lifecycle.Event.ON_RESUME)fun onResume() {Logger.d("onResume...")}@OnLifecycleEvent(Lifecycle.Event.ON_PAUSE)fun onPause() {Logger.d("onPause...")}@OnLifecycleEvent(Lifecycle.Event.ON_STOP)fun onStop() {Logger.d("onStop...")}@OnLifecycleEvent(Lifecycle.Event.ON_DESTROY)fun onDestroy() {Logger.d("onDestroy...")}
}
  • 方式三,实现 LifecycleEventObserver接口,在onStateChanged()方法中来处理不同类型的Event 类型。
class MainPresenter : LifecycleEventObserver {override fun onStateChanged(source: LifecycleOwner, event: Event) {when (event) {Event.ON_CREATE,Event.ON_START,Event.ON_RESUME,Event.ON_PAUSE,Event.ON_STOP,Event.ON_DESTROY -> Logger.d("onStateChanged...${event.name}")}}
}

3. 实现了 LifecycleObserver 接口之后,接下来调用 addObserver()方法添加我们的观察者,完成对生命周期的监听。

class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.layout_lifecycle)val presenter = MainPresenter()lifecycle.addObserver(presenter)}
}

到这里,我们就对架构组件 Lifecycle 的介绍就结束了,下一篇,来介绍架构组件中的 LiveData。

更多内容,可以订阅我的博客


参考链接

Lifecycle-Aware Components
Android 架构组件
了解 Activity 生命周期
Handling Lifecycles with Lifecycle-Aware Components

这篇关于Android 架构组件之 Lifecycle的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/746106

相关文章

C++ RabbitMq消息队列组件详解

《C++RabbitMq消息队列组件详解》:本文主要介绍C++RabbitMq消息队列组件的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. RabbitMq介绍2. 安装RabbitMQ3. 安装 RabbitMQ 的 C++客户端库4. A

PyQt6中QMainWindow组件的使用详解

《PyQt6中QMainWindow组件的使用详解》QMainWindow是PyQt6中用于构建桌面应用程序的基础组件,本文主要介绍了PyQt6中QMainWindow组件的使用,具有一定的参考价值,... 目录1. QMainWindow 组php件概述2. 使用 QMainWindow3. QMainW

Maven 插件配置分层架构深度解析

《Maven插件配置分层架构深度解析》:本文主要介绍Maven插件配置分层架构深度解析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Maven 插件配置分层架构深度解析引言:当构建逻辑遇上复杂配置第一章 Maven插件配置的三重境界1.1 插件配置的拓扑

Android NDK版本迭代与FFmpeg交叉编译完全指南

《AndroidNDK版本迭代与FFmpeg交叉编译完全指南》在Android开发中,使用NDK进行原生代码开发是一项常见需求,特别是当我们需要集成FFmpeg这样的多媒体处理库时,本文将深入分析A... 目录一、android NDK版本迭代分界线二、FFmpeg交叉编译关键注意事项三、完整编译脚本示例四

Android与iOS设备MAC地址生成原理及Java实现详解

《Android与iOS设备MAC地址生成原理及Java实现详解》在无线网络通信中,MAC(MediaAccessControl)地址是设备的唯一网络标识符,本文主要介绍了Android与iOS设备M... 目录引言1. MAC地址基础1.1 MAC地址的组成1.2 MAC地址的分类2. android与I

Android 实现一个隐私弹窗功能

《Android实现一个隐私弹窗功能》:本文主要介绍Android实现一个隐私弹窗功能,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 效果图如下:1. 设置同意、退出、点击用户协议、点击隐私协议的函数参数2. 《用户协议》、《隐私政策》设置成可点击的,且颜色要区分出来res/l

Android实现一键录屏功能(附源码)

《Android实现一键录屏功能(附源码)》在Android5.0及以上版本,系统提供了MediaProjectionAPI,允许应用在用户授权下录制屏幕内容并输出到视频文件,所以本文将基于此实现一个... 目录一、项目介绍二、相关技术与原理三、系统权限与用户授权四、项目架构与流程五、环境配置与依赖六、完整

Android 12解决push framework.jar无法开机的方法小结

《Android12解决pushframework.jar无法开机的方法小结》:本文主要介绍在Android12中解决pushframework.jar无法开机的方法,包括编译指令、框架层和s... 目录1. android 编译指令1.1 framework层的编译指令1.2 替换framework.ja

Android开发环境配置避坑指南

《Android开发环境配置避坑指南》本文主要介绍了Android开发环境配置过程中遇到的问题及解决方案,包括VPN注意事项、工具版本统一、Gerrit邮箱配置、Git拉取和提交代码、MergevsR... 目录网络环境:VPN 注意事项工具版本统一:android Studio & JDKGerrit的邮

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel