kotlin中SharedFlow的简单使用

事件总线配置示例

对于一次性事件通知(如Toast消息、导航指令),推荐使用无replay缓存且支持背压处理的配置

import kotlinx.coroutines.*

import kotlinx.coroutines.flow.*

// 事件总线配置:不缓存历史事件,缓冲区满时丢弃最旧事件

object AppEventBus {

private val _events = MutableSharedFlow<AppEvent>(

replay = 0, // 新订阅者不接收历史事件

extraBufferCapacity = 5, // 额外缓冲区容量

onBufferOverflow = BufferOverflow.DROP_OLDEST // 确保最新事件及时处理

)

val events: SharedFlow<AppEvent> = _events.asSharedFlow()

suspend fun postEvent(event: AppEvent) {

_events.emit(event)

}

}

// 应用事件类型

sealed class AppEvent {

data class ShowToast(val message: String) : AppEvent()

data class NavigationEvent(val destination: String) : AppEvent()

data class PermissionResult(val permission: String, val granted: Boolean) : AppEvent()

}

// 使用示例

class MainActivity {

fun setupEventListeners() {

lifecycleScope.launch {

repeatOnLifecycle(Lifecycle.State.STARTED) {

AppEventBus.events.collect { event ->

when (event) {

is AppEvent.ShowToast -> showToast(event.message)

is AppEvent.NavigationEvent -> navigateTo(event.destination)

is AppEvent.PermissionResult -> handlePermissionResult(event)

}

}

}

}

}

private fun triggerEvent() {

lifecycleScope.launch {

AppEventBus.postEvent(AppEvent.ShowToast("操作成功"))

AppEventBus.postEvent(AppEvent.NavigationEvent("home"))

AppEventBus.postEvent(AppEvent.PermissionResult("CAMERA", true))

}

}

此配置适用于一次性事件分发,确保新订阅者只接收订阅后的事件,同时通过DROP_OLDEST策略保证高优先级事件的处理。

UI状态管理配置

对于需要保持最新状态的UI组件(如页面加载状态、表单数据),推荐使用带replay缓存的配置。

import kotlinx.coroutines.flow.*

class UserProfileViewModel : ViewModel() {

// UI状态配置:缓存最新状态,确保新订阅者立即获得当前状态

private val _uiState = MutableSharedFlow<UserState>(

replay = 1, // 为新订阅者提供最新状态

extraBufferCapacity = 3, // 额外状态缓冲区

onBufferOverflow = BufferOverflow.SUSPEND // 状态必须完整传递

)

val uiState: SharedFlow<UserState> = _uiState.asSharedFlow()

// 用户状态数据类

data class UserState(

val isLoading: Boolean = false,

val userData: User? = null,

val errorMessage: String? = null

)

fun updateUserProfile() {

viewModelScope.launch {

// 发送加载状态

_uiState.emit(UserState(isLoading = true))

// 模拟网络请求

delay(1000)

// 发送成功状态

_uiState.emit(UserState(

isLoading = false,

userData = User(name = "张三", email = "zhangsan@example.com")

))

}

}

}

这种配置保证新进入的UI组件能立即获得当前应用状态,避免状态不一致问题。

高频率数据流配置

对于传感器数据、实时日志等高频数据流,需采用优化的背压处理策略:

import kotlinx.coroutines.flow.*

class SensorDataProcessor {

// 高频数据流配置:快速处理最新数据,避免积压

private val _sensorFlow = MutableSharedFlow<SensorData>(

replay = 0, // 不缓存历史数据

extraBufferCapacity = 10, // 较大的缓冲区

onBufferOverflow = BufferOverflow.DROP_LATEST // 丢弃最新数据,保留处理中的序列

)

val sensorFlow: SharedFlow<SensorData> = _sensorFlow.asSharedFlow()

// 模拟高频数据产生

fun startProducingData() {

viewModelScope.launch {

var count = 0

while (true) {

_sensorFlow.emit(SensorData(

value = Math.random(),

timestamp = System.currentTimeMillis()

))

count++

delay(10) // 每10毫秒产生一个数据点

}

}

}

}

// 传感器数据结构

data class SensorData(

val value: Double,

val timestamp: Long

)

DROP_LATEST策略适用于消费者处理速度跟不上生产速度的场景,确保系统稳定性。

多订阅者广播配置

当需要向多个组件广播相同数据时(如埋点上报、全局配置更新),可采用以下配置:

import kotlinx.coroutines.flow.*

class GlobalConfigManager {

// 多订阅者配置:缓存配置历史,支持批量订阅

private val _configFlow = MutableSharedFlow<ConfigUpdate>(

replay = 5, // 缓存最近5次配置变更

extraBufferCapacity = 20, // 支持大量订阅者

onBufferOverflow = BufferOverflow.DROP_OLDEST // 配置更新以最新为准

)

val configFlow: SharedFlow<ConfigUpdate> = _configFlow.asSharedFlow()

// 批量更新配置

suspend fun updateMultipleConfigs(configs: List<ConfigUpdate>) {

configs.forEach { config ->

_configFlow.emit(config)

}

}

}

// 配置更新事件

data class ConfigUpdate(

val key: String,

val value: Any,

val version: Int

)

这种配置特别适合需要追踪配置变更历史的场景,新订阅者可获取最近的配置变更记录。

总结

SharedFlow的配置应根据具体业务需求灵活调整:事件总线适合无缓存配置,UI状态管理需要replay缓存,高频数据流需优化背压处理,多订阅者场景则需要更大的缓冲容量。合理配置这三个参数能够有效平衡性能与功能需求。

相关推荐
小陈工36 分钟前
Python安全编程实践:常见漏洞与防护措施
运维·开发语言·人工智能·python·安全·django·开源
majingming1236 小时前
FUNCTION
java·前端·javascript
zopple6 小时前
常见的 Spring 项目目录结构
java·后端·spring
是娇娇公主~7 小时前
C++ 中 std::deque 的原理?它内部是如何实现的?
开发语言·c++·stl
SuperEugene7 小时前
Axios 接口请求规范实战:请求参数 / 响应处理 / 异常兜底,避坑中后台 API 调用混乱|API 与异步请求规范篇
开发语言·前端·javascript·vue.js·前端框架·axios
xuxie998 小时前
N11 ARM-irq
java·开发语言
cjy0001118 小时前
springboot的 nacos 配置获取不到导致启动失败及日志不输出问题
java·spring boot·后端
wefly20179 小时前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
zhenxin01229 小时前
Spring Boot实现定时任务
java
小江的记录本9 小时前
【事务】Spring Framework核心——事务管理:ACID特性、隔离级别、传播行为、@Transactional底层原理、失效场景
java·数据库·分布式·后端·sql·spring·面试