Android---MVP 中 presenter 声明周期的管理
我们经常在 Android MVP 架构中的 Presenter 层做一些耗时操作,比如请求网络数据,然后根据请求后的结果刷新 View。但是,如果按返回结束 Activity,而 Presenter 依然在执行耗时操作。那么就有可能造成内存泄漏,严重时甚至会造成程序崩溃。因为 Presenter 中的 View 已经变为 null。
为了解决这个问题,需要将 Activity 的某些生命周期方法与 Presenter 保持一致。
Lifecycle 绑定 presenter 生命周期
Lifecycle 的使用很简单,Activity 通过继承 AppCompatActivity 会自动继承来自父类 ComponentActivity 的方法 getLifeCycle。具体如下

onStateChanged 方法会在 Activity 的生命周期发生变化时被触发,比如当我们打开 LoginActivity 时就会显示如下日志

当中 LoginActivity 界面按下返回键,则打印如下内容

Lifecycle 还提供了注解的方式使用,因此我们可以很容易创建一个接口 IPresenter。在这个接口中声明对各种 Activity 声明周期的回调,如下所示

上图中 IPresenter 接口通过注解的方式将 Activity 的声明周期绑定到相应的方法上。我们只需要在 BsetPresenter 中实现上述方法,并在方法中数据绑定与取消的操作即可,具体如下

注意:上图中代码存在一些问题,使用了 Android 中的 Log 来打印日志信息。严格来说,在 Presenter 层应该禁止出现任何 Android 中的类。这里为了快速演示效果,所以直接使用 Log 打印日志。
接下来,修改 LoginActivity,将 BasePresenter 注册到 LifeCycle 中即可,如下代码所示

从新打开 LoginActivity,显示日志如下

关闭 LoginActivity,显示日志如下

可以看出,当 Activity 执行 onDestory() 时,BasePresenter 的 onDestory() 方法也会被执行。
在 LoginActivity 方法中有 login() 方法,此方法会执行 BasePresenter 中的 login() 方法。如下代码

在 BasePresenter 中的 login() 方法中模拟执行了一段耗时操作。如果在 Activity onDestory() 时,BasePresenter 还没有处理完耗时操作,则会造成内存泄漏。
解决办法就是在 BasePresenter 的 onDestory 方法中停止正在执行的耗时操作。如下所示

合理使用 Presenter 生命周期
并不是所有的 Activity 的生命周期都需要通知 Presenter。举个例子,假如我们需求是根据 GPS 定位,展示用户的位置。但是为了节省电量,有可能会在灭屏之后,解绑定 GPS 定位的接收事件。
如果使用 MVP 架构,需要有一个 TrackingActivity 实现 MVP 的接口 TrackingView,并在生命周期方法中调用 presenter 的相应方法。如下所示

TrackingPresenter 是 presenter 层的实现,内部实现了 GPS 定位的监听事件,并分别在 resume 和 stop 方法中绑定和解绑定 GPS。如下

上述写法是常规的 MVP 写法,但是存在两个问题:1)GpsTracker 实际的控制周期是跟 Activity 有关的,因为亮屏和灭屏事件是在 Activity 中接收的。中间多了一层 Presenter 层其实是多余的;2)从重构的角度看,TrackingPresenter 其实违反了职责单一原则(Single Responsibility)。因为 Presenter 层的主要作用是用来刷新 View,但是上述代码中的 TrackingPresenter 还负责对 GpsTracker 进行管理。
这种情况下,我们可以将 GpsTracker 初始化在 Activity 中,将 GpsTracker 的绑定与解绑定都在 Activity 中管理,最后将 GpsTracker 传给 TrackingPresenter 执行业务上的逻辑。具体实现如下

TrackingActivity 中对 tracker 进行管理,并且根据 Presenter 层的逻辑处理,回调 showCurrentPosition() 方法。
TrackingPresenter
TrackingPresenter 只负责对 Gps 事件的监听并根据结果刷新 View。这样 View 层和 Presenter 层的职责单一原则就完咋。在完成实际需求的前提下,也丝毫不影响 Presenter 层的单元测试

当然,并没有绝对正确或错误的架构,说到底代码具体要怎样写、功能具体应该怎样实现,最终还是要看实际业务场景。
总结
主要对 MVP 架构中 Presenter 层的使用做了2点优化介绍:
1. 如何支持 Presenter 的生命周期,使其在 Activity 被销毁时也能取消相应的耗时请求。
2. 合理使用 Presenter 的生命周期,Activity 中所有的方法都委托给 Presenter 来处理是不合理的。这样会造成 Presenter 层极其庞大,也难以维护,有时也会违反职责单一原则。
相关文章:
Android---MVP 中 presenter 声明周期的管理
我们经常在 Android MVP 架构中的 Presenter 层做一些耗时操作,比如请求网络数据,然后根据请求后的结果刷新 View。但是,如果按返回结束 Activity,而 Presenter 依然在执行耗时操作。那么就有可能造成内存泄漏,严重时甚…...
Oracle中的索引碎片
索引碎片是指索引在存储空间上不连续的分布情况,它可能会影响到数据库性能和查询效率。索引碎片化主要由以下几个原因导致: 插入、更新和删除操作:当对表中的数据进行插入、更新或删除操作时,索引也需要相应地更新。这些DML操作可…...
Java必刷入门递归题×5(内附详细递归解析图)
目录 1.求N的阶乘 2.求12...N的和 3.顺序打印数字的每一位 4.求数字的每一位之和 5.求斐波拉契数列 1.求N的阶乘 (1)解析题目意思 比如求5的阶乘,符号表示就是5!;所以5!5*4*3*2*1我们下面使用简单的…...
android 闪屏图适配尺寸
不同的 Android 设备可能具有不同的屏幕尺寸和分辨率,因此最好提供不同尺寸的启动画面图像,以确保与各种设备的兼容性。 以下是 Android 启动画面图像的一些最常见尺寸: 320 x 480像素(肖像) 480 x 320像素࿰…...
正则表达式中(?s)与(?m)的区别
理论: (?m) 和 (?s) 是正则表达式中的两个模式标志,它们具有不同的作用: (?m) 多行模式标志(也称为 “multiline” 模式): 默认情况下,正则表达式将整个输入字符串视为单行多行文本中使用…...
Clickhouse学习笔记(11)—— 数据一致性
使用合并树引擎时,无论是ReplacingMergeTree还是SummingMergeTree,都只能保证数据的最终一致性,因为数据的去重、聚合等操作会在数据合并的期间进行,而合并会在后台以一个不确定的时间进行,因此无法预先计划࿱…...
【uniapp】六格验证码输入框实现
效果图 代码实现 <view><view class"tips">已发送验证码至<text class"tips-phone">{{ phoneNumber }}</text></view><view class"code-input-wrap"><input class"code-input" v-model"…...
【react hook】在react hook组件中,使用Antd Table组件,columns数据异步获取,list数据更新但没有rerender解决办法
情景描述 我们有一个react组件,显示了一个Antd Table组件,设置了一个columns变量并复制给Table的columns属性,由于我们请求的datasource来源是异步的,示例伪代码如下: const [columns, setColumns] useState([]); /…...
ChatGPT的图识别来了
前几天ChatGPT推出了Dall-E 3功能,可以根据文字和描述一段话来生成一个或者一组图。 这次又来重磅了,图识别又来了!换句话说,也即是文生图,图生文都可以实现了,一起来试试 1、解释图中的意思 ࿰…...
java Stream编程笔记
文章目录 Stream介绍什么是 Stream? Stream中间操作过滤操作(filter)映射操作(map)排序操作(sorted)截断操作(limit 和 skip) Stream 的终止操作forEach 和 peek聚合操作…...
顶顶通语音识别使用说明
介绍 顶顶通语音识别软件(asrproxy)是一个对接了多种语音识别接口的语音识别系统。可私有化部署(支持中文英文和方言等,支持一句话识别、实时流识别、多声道录音文件识别。 原理 asrproxy内嵌了阿里达摩院的开源语音识别工具包FunASR,后续我们也会使用自有的预料…...
重磅发布 OpenAI 推出用户自定义版 ChatGPT
文章目录 重磅发布 OpenAI 推出用户自定义版 ChatGPT个人简介 重磅发布 OpenAI 推出用户自定义版 ChatGPT OpenAI 首届开发者大会 (OpenAI DevDay) 于北京时间 11 月 7 日凌晨 02:00 开始,大会上宣布了一系列平台更新。其中一个重要更新是用户可以创建他们自己的自定…...
Java 幼儿园(20231111)读取 json 文件
1、功能场景 (1)多人合作开发一个功能模块时,需要调用外部接口 (2)对方接口的开发工作还没有完成,只能提供一个返回值的示例文件 json 文件。 (3)返回的 json 数据多达几百个字段。 …...
云计算、大数据技术的智慧工地,实现对建筑工地实时监测、管理和控制的一种新型建筑管理方式
智慧工地是利用物联网、云计算、大数据等技术,实现对建筑工地实时监测、管理和控制的一种新型建筑管理方式。 智慧工地架构: 1、终端层: 充分利用物联网技术、移动应用、智能硬件设备提高现场管控能力。通过RFID、传感器、摄像头、手机等终…...
功能案例 -- 通过开关,改变白天和黑夜
效果展示 代码展示 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><style>:root {--default-bac-color: #f…...
Linux编写一个极简版本的Shell
Linux编写一个极简版本的Shell 📟作者主页:慢热的陕西人 🌴专栏链接:Linux 📣欢迎各位大佬👍点赞🔥关注🚓收藏,🍉留言 本博客主要内容在Linux环境下ÿ…...
亚马逊云AI应用科技创新下的Amazon SageMaker使用教程
目录 Amazon SageMaker简介 Amazon SageMaker在控制台的使用 模型的各项参数 pytorch训练绘图部分代码 Amazon SageMaker简介 亚马逊SageMaker是一种完全托管的机器学习服务。借助 SageMaker,数据科学家和开发人员可以快速、轻松地构建和训练机器学习模型&#…...
Eigen:旋转向量(Angle-Axis)转换为四元素和旋转矩阵
0. 在固定欧拉角系下。 绕固定系旋转,旋转的先后顺序为X、Y、Z。当然也支持XYZ的任意顺序旋转。 1. 转为四元素 Eigen::Quaterniond q Eigen::AngleAxisd(yaw, Eigen::Vector3d::UnitZ()) *Eigen::AngleAxisd(pitch, Eigen::Vector3d::UnitY()) *Eigen::AngleAxi…...
C#8.0本质论第十二章--泛型
C#8.0本质论第十二章–泛型 C#通过泛型来促进代码重用,在词义上等价于C模板。 在泛型编程中,数据类型也是一种参数。 12.1如果C#没有泛型 为object的方法使用值类型时,“运行时”将自动对它进行装箱,获取值类型的实例时则需要…...
Python与ArcGIS系列(七)自动化打印地图
目录 0 简述1 获取可用打印机列表2 打印地图3 导出地图至PDF4 导出地图至图像0 简述 本篇介绍如何利用arcpy实现获取可用打印机列表、打印地图、导出地图至PDF和图像。 1 获取可用打印机列表 通过arcpy提供的ListPrinterNames()函数可以生成可用的打印机列表。 import arcpy.m…...
LiuJuan Z-Image Generator惊艳效果:低光环境人像噪点控制与细节保留
LiuJuan Z-Image Generator惊艳效果:低光环境人像噪点控制与细节保留 你有没有试过在光线不足的环境下拍照?拍出来的照片是不是经常噪点满天飞,人脸细节糊成一团,后期怎么修都救不回来?对于摄影师和内容创作者来说&am…...
TikTok爆火C语言创意:电脑无硬件发无线电,靠谱吗?
一、刷爆TikTok的技术神操作,无硬件也能发无线电? 2026年3月17日,有一条C语言创意短视频,它刷爆了TikTok,在单日的时候,斩获了10万以上的播放量,以及5万以上的点赞量。并且它登顶了当日C语言创…...
Tessent Boundary Scan: Revolutionizing PCB Testing with Embedded DFT Solutions
1. 边界扫描技术如何改变PCB测试格局 十年前我第一次接触PCB测试时,车间里还堆满了密密麻麻的测试针床。老师傅们需要手动调整数百个探针位置,稍有不慎就会造成误测。随着表面贴装器件(SMD)的普及,这种传统在线测试(In-circuit test)的局限性…...
使用VSCode远程开发并调试Qwen3.5-4B模型调用代码
使用VSCode远程开发并调试Qwen3.5-4B模型调用代码 1. 前言:为什么需要远程开发? 当你开始接触大模型开发时,可能会遇到一个常见问题:本地电脑性能不足,无法流畅运行像Qwen3.5-4B这样的模型。这时候,远程开…...
Chandra OCR科研复现教程:olmOCR基准测试环境搭建与83.1分结果验证
Chandra OCR科研复现教程:olmOCR基准测试环境搭建与83.1分结果验证 4 GB显存即可运行,83分OCR精度,表格/手写/公式一次搞定,输出直接是Markdown 1. 项目背景与核心价值 Chandra是Datalab.to在2025年10月开源的"布局感知&quo…...
加密货币钱包原理与开发
加密货币钱包原理与开发:数字资产的安全之门 在区块链技术蓬勃发展的今天,加密货币钱包作为管理数字资产的核心工具,已成为用户进入加密世界的必备钥匙。无论是比特币、以太坊还是其他代币,钱包不仅存储密钥,更是实现…...
对未来十年技术发展的预测
未来十年技术发展:颠覆与重构的黄金时代 科技的迭代速度正以指数级增长,未来十年或将迎来人类历史上最具颠覆性的技术变革。从人工智能的自我进化到量子计算的实用化突破,从生物科技的基因重塑到能源技术的零碳革命,技术边界将被…...
ESP8266驱动1.44英寸ST7735 TFT屏的实战指南与图像显示优化
1. ESP8266与ST7735屏的硬件连接实战 第一次用ESP8266驱动1.44寸ST7735屏时,最让我头疼的就是引脚接线问题。不同厂商的屏幕引脚定义可能略有差异,但核心信号线基本一致。我手头这块屏采用8针SPI接口,实际测试发现用NodeMCU开发板连接最方便。…...
阿里通义Z-Image-GGUF体验:中英文提示词生成精美图片实测
阿里通义Z-Image-GGUF体验:中英文提示词生成精美图片实测 1. 模型概述与特点 1.1 什么是Z-Image-GGUF Z-Image-GGUF是阿里巴巴通义实验室开源的一款文生图AI模型,基于GGUF量化技术优化,能够在较低显存环境下实现高质量的图像生成。该模型支…...
uniapp结合微信公众号H5静默授权:从本地调试到获取openid的完整实践
1. 理解静默授权与openid的核心逻辑 静默授权是微信公众号开发中获取用户基础信息的常用方式,特别适合不需要用户主动授权的场景。与需要用户点击确认的snsapi_userinfo授权不同,snsapi_base授权可以在用户无感知的情况下完成。这个过程中最关键的产出物…...
