当前位置: 首页 > article >正文

Android学习之响应式编程

本篇基于DeepSeek 搜索结果修改。

一、响应式编程基础认知

1.1 为什么需要响应式编程?

在传统的Android开发中,我们经常会遇到以下痛点:

// 传统方式处理数据变化
button.setOnClickListener {// 触发网络请求fetchDataFromNetwork { result ->// 更新UItextView.text = result// 如果数据依赖其他状态,需要手动管理if (result.isNotEmpty()) {recyclerView.visibility = View.VISIBLE} else {recyclerView.visibility = View.GONE}}
}

这种方式存在以下问题:

  • 多层嵌套导致代码可读性差
  • 手动管理状态变化容易出错
  • 线程切换复杂
  • 难以处理异步数据流

1.2 响应式编程核心概念

1.2.1 数据流

数据流是响应式编程的核心,它可以是:

  • 同步数据流:如集合
  • 异步数据流:如网络请求、传感器数据
1.2.2 观察者模式

观察者模式由以下部分组成:

  • 被观察对象(Observable)
  • 观察者(Observer)
  • 订阅关系(Subscription)
// 简化版观察者模式实现
class Observable<T> {private val observers = mutableListOf<Observer<T>>()fun subscribe(observer: Observer<T>) {observers.add(observer)}fun emit(value: T) {observers.forEach { it.onNext(value) }}
}interface Observer<T> {fun onNext(value: T)fun onError(error: Throwable)fun onComplete()
}
1.2.3 操作符

操作符用于转换、过滤和组合数据流:

  • map:转换数据
  • filter:过滤数据
  • flatMap:处理嵌套数据流
  • debounce:防抖处理

二、Android响应式编程工具

2.1 LiveData入门

2.1.1 LiveData基础使用

LiveData是Android官方提供的可观察数据持有者类:

// 创建LiveData
val liveData = MutableLiveData<String>()// 观察LiveData
liveData.observe(this) { value ->// 数据变化时更新UItextView.text = value
}// 更新LiveData
liveData.value = "New Value"
2.1.2 LiveData与ViewModel结合

LiveData通常与ViewModel结合使用:

class MyViewModel : ViewModel() {private val _data = MutableLiveData<String>()val data: LiveData<String> = _datafun loadData() {// 模拟数据加载viewModelScope.launch {delay(1000)_data.value = "Loaded Data"}}
}

2.2 RxJava深入

2.2.1 RxJava基本概念

RxJava使用Observable和Observer处理数据流:

// 创建Observable
Observable<String> observable = Observable.just("Hello", "RxJava", "World");// 创建Observer
Observer<String> observer = new Observer<String>() {@Overridepublic void onSubscribe(Disposable d) {// 订阅时调用}@Overridepublic void onNext(String s) {// 接收到数据时调用Log.d("RxJava", s);}@Overridepublic void onError(Throwable e) {// 发生错误时调用}@Overridepublic void onComplete() {// 完成时调用}
};// 订阅
observable.subscribe(observer);
2.2.2 RxJava操作符实战

使用操作符处理复杂数据流:

Observable.just("apple", "banana", "cherry").map(fruit -> fruit.toUpperCase()).filter(fruit -> fruit.startsWith("B")).subscribe(fruit -> {Log.d("RxJava", "Filtered: " + fruit);});

2.3 Kotlin Flow实战

2.3.1 Flow基础

Flow是Kotlin协程中的响应式编程库:

// 创建Flow
fun numbers(): Flow<Int> = flow {for (i in 1..3) {delay(100)emit(i)}
}// 收集Flow
viewModelScope.launch {numbers().map { it * it }.collect { value ->Log.d("Flow", "Received: $value")}
}
2.3.2 Flow与网络请求

使用Flow处理网络请求:

suspend fun fetchData(): Flow<Data> = flow {// 模拟网络请求val response = apiService.getData()emit(response)
}

三、实战案例进阶

3.1 搜索功能实现对比

3.1.1 传统实现方式
editText.addTextChangedListener(object : TextWatcher {override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {}override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {// 每次文本变化都触发搜索performSearch(s.toString())}override fun afterTextChanged(s: Editable?) {}
})private fun performSearch(query: String) {// 执行搜索
}
3.1.2 使用RxJava实现防抖搜索
RxTextView.textChanges(editText).debounce(300, TimeUnit.MILLISECONDS).filter(text -> text.length() > 2).switchMap(query -> searchApi.search(query.toString()).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread())).subscribe(results -> {// 更新UI});
3.1.3 使用Flow实现防抖搜索
editText.textChanges().debounce(300).filter { it.length > 2 }.mapLatest { query ->repository.search(query.toString())}.flowOn(Dispatchers.IO).onEach { results ->// 更新UI}.launchIn(lifecycleScope)

3.2 多数据源合并

3.2.1 合并本地和远程数据
fun getUsers(): Flow<List<User>> = flow {// 先发射本地数据emit(localDataSource.getUsers())// 再发射远程数据val remoteUsers = remoteDataSource.getUsers()localDataSource.saveUsers(remoteUsers)emit(remoteUsers)
}

四、总结

可以看出,Android 的响应式编程范式主要是由观察者模式结合响应式流来实现的。观察者模式作为核心架构,构建起数据生产者与消费者之间的订阅关系,使得数据变化能够及时被关注;响应式流则负责承载数据的流动与处理,通过操作符对数据进行转换、过滤等操作,实现复杂业务逻辑的编排。

相关文章:

Android学习之响应式编程

本篇基于DeepSeek 搜索结果修改。 一、响应式编程基础认知 1.1 为什么需要响应式编程&#xff1f; 在传统的Android开发中&#xff0c;我们经常会遇到以下痛点&#xff1a; // 传统方式处理数据变化 button.setOnClickListener {// 触发网络请求fetchDataFromNetwork { res…...

如何使用极狐GitLab 软件包仓库功能托管 helm chart?

极狐GitLab 是 GitLab 在中国的发行版&#xff0c;关于中文参考文档和资料有&#xff1a; 极狐GitLab 中文文档极狐GitLab 中文论坛极狐GitLab 官网 软件包库中的 Helm charts (BASIC ALL) WARNING:Helm chart 库正在开发中&#xff0c;由于功能有限&#xff0c;尚未准备好用…...

中国古代史4

东汉 公元25年&#xff0c;刘秀建立东汉&#xff0c;定都洛阳&#xff0c;史称光武中兴 白马寺&#xff1a;汉明帝时期建立&#xff0c;是佛教传入中国后兴建的第一座官办寺院&#xff0c;有中国佛教的“祖庭”和“释源”之称&#xff0c;距今1900多年历史 班超—西域都护—投…...

Nacos源码—8.Nacos升级gRPC分析六

大纲 7.服务端对服务实例进行健康检查 8.服务下线如何注销注册表和客户端等信息 9.事件驱动架构源码分析 一.处理ClientChangedEvent事件 也就是同步数据到集群节点&#xff1a; public class DistroClientDataProcessor extends SmartSubscriber implements DistroDataSt…...

基于Vue3.0的高德地图api教程005:实现绘制线并编辑功能

文章目录 6、绘制多段线6.1 绘制多段线6.1.1 开启绘制功能6.1.2 双击完成绘制6.1.3 保存到数据库6.2 修改多段线6.2.1 点击线,进入编辑模式6.2.2 编辑线6.3 完整代码6、绘制多段线 6.1 绘制多段线 6.1.1 开启绘制功能 实现代码: const changeSwitchDrawPolyline = ()=>…...

SpringBoot 自动装配原理 自定义一个 starter

目录 1、pom.xml 文件1.1、parent 模块1.1.1、资源文件1.1.1.1、resources 标签说明1.1.1.2、从 Maven 视角&#xff1a;资源处理全流程​ 1.1.2、插件 1.2、dependencies 模块 2、启动器3、主程序3.1、SpringBootApplication 注解3.2、SpringBootConfiguration 注解3.2.1、Con…...

【C++进阶篇】多态

深入探索C多态&#xff1a;静态与动态绑定的奥秘 一. 多态1.1 定义1.2 多态定义及实现1.2.1 多态构成条件1.2.1.1 实现多态两个必要条件1.2.1.2 虚函数1.2.1.3 虚函数的重写/覆盖1.2.1.4 协变1.2.1.5 析构函数重写1.2.1.6 override和final关键字1.2.1.7 重载/重写/隐藏的对⽐ 1…...

Redis 基础详解:从入门到精通

在当今互联网应用开发领域&#xff0c;数据存储与处理的性能和效率至关重要。Redis&#xff08;Remote Dictionary Server&#xff09;作为一款开源的、基于内存的键值存储系统&#xff0c;凭借其出色的性能和丰富的功能&#xff0c;被广泛应用于数据库、缓存、消息中间件等场景…...

Android Studio的jks文件

在 Android Studio 中&#xff0c;.jks 文件是 Java KeyStore&#xff08;Java 密钥库&#xff09;文件的一种&#xff0c;用于存储和管理用于签署 Android 应用程序的数字证书和私钥。 一、.jks 文件的作用 在 Android 开发中&#xff0c;.jks 文件主要用于&#xff1a; 应用…...

互联网大厂Java面试实战:从Spring Boot到微服务的技术问答与解析

&#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精通 &#x1f601; 2. 毕业设计专栏&#xff0c;毕业季咱们不慌忙&#xff0c;几百款毕业设计等你选。 ❤️ 3. Python爬虫专栏…...

《AI大模型应知应会100篇》第60篇:Pinecone 与 Milvus,向量数据库在大模型应用中的作用

第60篇&#xff1a;Pinecone与Milvus&#xff0c;向量数据库在大模型应用中的作用 摘要 本文将系统比较Pinecone与Milvus两大主流向量数据库的技术特点、性能表现和应用场景&#xff0c;提供详细的接入代码和最佳实践&#xff0c;帮助开发者为大模型应用选择并优化向量存储解…...

HDFS客户端操作

一、命令行工具操作 HDFS 命令行工具基于 hdfs dfs 命令&#xff0c;语法类似 Linux 文件操作。 1. 文件操作 bash # 创建目录 hdfs dfs -mkdir /test# 递归创建多级目录 hdfs dfs -mkdir -p /test/data/logs# 上传本地文件到 HDFS hdfs dfs -put local_file.txt /test/# 从…...

MySQL--视图详解

介绍 视图&#xff08;View&#xff09;是一种虚拟存在的表。视图中的数据并不在数据库中实际存在&#xff0c;行和列数据来自定义视图的查询中使用的表&#xff08;称为基表&#xff09;&#xff0c;并且是在使用视图时动态生成的。 简而言之&#xff1a;视图只保存了查询的…...

Java学习手册:客户端负载均衡

一、客户端负载均衡的概念 客户端负载均衡是指在客户端应用程序中&#xff0c;根据一定的算法和策略&#xff0c;将请求分发到多个服务实例上。与服务端负载均衡不同&#xff0c;客户端负载均衡不需要通过专门的负载均衡设备或服务&#xff0c;而是直接在客户端进行请求的分发…...

Docker私有仓库实战:官方registry镜像实战应用

抱歉抱歉&#xff0c;离职后反而更忙了&#xff0c;拖了好久&#xff0c;从4月拖到现在&#xff0c;在学习企业级方案Harbor之前&#xff0c;我们先学习下官方方案registry&#xff0c;话不多说&#xff0c;详情见下文。 注意&#xff1a;下文省略了基本认证 TLS加密&#xff…...

LeetCode 373 查找和最小的 K 对数字题解

LeetCode 373 查找和最小的 K 对数字题解 题目描述 给定两个以升序排列的整数数组 nums1 和 nums2&#xff0c;以及一个整数 k。定义一对值 (u,v)&#xff0c;其中第一个元素来自 nums1&#xff0c;第二个元素来自 nums2。请找到和最小的 k 个数对。 解题思路 最小堆优化法…...

WebSocket集成方案对比

​ WebSocket集成方案对比与实战 架构选型全景图 #mermaid-svg-BEuyOkkoP6cFygI0 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-BEuyOkkoP6cFygI0 .error-icon{fill:#552222;}#mermaid-svg-BEuyOkkoP6cFygI0 .er…...

深入理解 Istio v1.25.2

要深入理解 Istio 的最新版本&#xff08;截至 2025 年 5 月&#xff0c;最新版本为 1.25.2&#xff0c;发布Iweb:1⁊&#xff09;源码&#xff0c;我们可以通过分析其核心组件和代码结构来加深对 Istio 的理解。以下是对 Istio 源码的解读&#xff0c;结合其架构和功能&#x…...

使用conda导致无法找到libpython动态库

最近在用 AFL 的时候编译完成后遇到如下的报错&#xff1a; afl-fuzz: error while loading shared libraries: libpython3.9.so.1.0: cannot open shared object file: No such file or directory然后发现是因为编译时用的Python环境是通过miniconda构建的虚拟环境&#xff0…...

Redis+Caffeine构建高性能二级缓存

大家好&#xff0c;我是摘星。今天为大家带来的是RedisCaffeine构建高性能二级缓存&#xff0c;废话不多说直接开始~ 目录 二级缓存架构的技术背景 1. 基础缓存架构 2. 架构演进动因 3. 二级缓存解决方案 为什么选择本地缓存&#xff1f; 1. 极速访问 2. 减少网络IO 3…...

MyBatis-Plus使用 wrapper.apply() 添加自定义 SQL 片段

在 MyBatis-Plus 中&#xff0c;wrapper.apply() 方法允许你在构建查询条件时插入任意的 SQL 片段。这对于实现一些复杂的查询需求特别有用&#xff0c;比如添加子查询、使用数据库特定函数等&#xff1b; 示例 1: 基本应用 import com.baomidou.mybatisplus.core.conditions…...

【计算机网络】NAT技术、内网穿透与代理服务器全解析:原理、应用及实践

&#x1f4da; 博主的专栏 &#x1f427; Linux | &#x1f5a5;️ C | &#x1f4ca; 数据结构 | &#x1f4a1;C 算法 | &#x1f152; C 语言 | &#x1f310; 计算机网络 上篇文章&#xff1a;以太网、MAC地址、MTU与ARP协议 下篇文章&#xff1a;五种IO模型与阻…...

Python训练打卡Day21

常见的降维算法&#xff1a; # 先运行预处理阶段的代码 import pandas as pd import pandas as pd #用于数据处理和分析&#xff0c;可处理表格数据。 import numpy as np #用于数值计算&#xff0c;提供了高效的数组操作。 import matplotlib.pyplot as plt #用于绘…...

【大模型MCP协议】MCP官方文档(Model Context Protocol)一、开始——1. 介绍

https://modelcontextprotocol.io/tutorials/building-mcp-with-llms 文章目录 介绍为什么选择MCP&#xff1f;总体架构 开始使用快速入门示例 教程探索MCP贡献支持和反馈探索 MCP贡献代码支持与反馈 介绍 开始使用模型上下文协议&#xff08;MCP&#xff09; C# SDK已发布&…...

三大告警方案解析:从日志监控到流处理的演进之路

引言&#xff1a;告警系统的核心挑战与演进逻辑 在分布式系统中&#xff0c;实时告警是实现业务稳定性的第一道防线。随着系统复杂度提升&#xff0c;告警机制从简单的日志匹配逐步演进到流式处理的秒级响应。本文将基于‌三大主流方案‌&#xff08;日志告警、离线统计、实时流…...

node .js 启动基于express框架的后端服务报错解决

问题&#xff1a; node .js 用npm start 启动基于express框架的后端服务报错如下&#xff1a; /c/Program Files/nodejs/npm: line 65: 26880 Segmentation fault "$NODE_EXE" "$NPM_CLI_JS" "$" 原因分析&#xff1a; 遇到 /c/Program F…...

互联网大厂Java求职面试实战:Spring Boot与微服务场景深度解析

&#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精通 &#x1f601; 2. 毕业设计专栏&#xff0c;毕业季咱们不慌忙&#xff0c;几百款毕业设计等你选。 ❤️ 3. Python爬虫专栏…...

并发笔记-信号量(四)

文章目录 背景与动机31.1 信号量&#xff1a;定义 (Semaphores: A Definition)31.2 二元信号量 (用作锁) (Binary Semaphores - Locks)31.3 用于排序的信号量 (Semaphores For Ordering)31.4 生产者/消费者问题 (The Producer/Consumer (Bounded Buffer) Problem)31.5 读写锁 (…...

【HTOP 使用指南】:如何理解主从线程?(以 Faster-LIO 为例)

htop 是 Linux 下常用的进程监控工具&#xff0c;它比传统的 top 更友好、更直观&#xff0c;尤其在分析多线程或多进程程序时非常有用。 以下截图就是在运行 Faster-LIO 实时建图时的 htop 状态展示&#xff1a; &#x1f50d; 一、颜色说明 白色&#xff08;或亮色&#xf…...

数据同步DataX任务在线演示

数据同步DataX任务在线演示 1. 登录系统 访问系统登录页面&#xff0c;输入账号密码完成身份验证。 2. 环境准备 下载datax安装包&#xff0c;并解压到安装目录 3. 集群创建 点击控制台-多集群管理 计算组件添加DataX 配置DataX引擎,Datax.local.path填写安装目录。 4. …...