Kotlin Flow 的 emit 和 tryEmit 有什么区别 ?

在 Kotlin Flow 中,emit()tryEmit() 都用于向 Flow 发送值,但它们的关键区别在于背压处理协程上下文

一、主要区别

1. emit()

  • 挂起函数:会挂起直到有空间接收值(处理背压)
  • 协程安全:只能在协程作用域中调用
  • 推荐使用 :在 collectcallbackFlow 中优先使用
kotlin 复制代码
flow {
    // 会挂起直到接收方准备好
    emit(1)
    emit(2)
}

2. tryEmit()

  • 非挂起函数 :尝试立即发射值,如果接收方未准备好则返回 false
  • 可能丢数据 :返回 false 时值会被丢弃
  • 有限场景:主要在 MutableStateFlow 和 MutableSharedFlow 中使用
kotlin 复制代码
val _state = MutableStateFlow(0)

fun updateValue(newValue: Int) {
    // 尝试发射,如果失败则丢弃
    val success = _state.tryEmit(newValue)
    if (!success) {
        // 处理发射失败的情况
    }
}

二、使用场景对比

1. 使用 emit() 的场景

kotlin 复制代码
// 1. 在 flow {} 构建器中
flow {
    for (i in 1..10) {
        delay(100)
        emit(i)  // ✅ 正确
    }
}

// 2. 在 callbackFlow 中
callbackFlow {
    callback.setOnData { data ->
        trySend(data)  // callbackFlow 特殊方法
    }
    awaitClose { callback.removeListener() }
}

2. 使用 tryEmit() 的场景

kotlin 复制代码
// 主要在 MutableStateFlow/MutableSharedFlow 中
class ViewModel {
    private val _uiState = MutableStateFlow<UiState>(Loading)
    val uiState: StateFlow<UiState> = _uiState
    
    fun update() {
        // 非挂起函数中尝试更新状态
        if (!_uiState.tryEmit(Success(data))) {
            // 处理背压(通常日志记录)
        }
    }
}

3. 下面是一个小表格,可以帮助快速了解何时使用 emittryEmit

场景 使用 emit 使用 tryEmit
在挂起函数内部 避免使用(除非非常紧急)
UI 点击或生命周期事件 可能(如果需要快速更新)
回调、监听器或后台线程 避免使用(不能挂起)
需要保证数据送达 否(需自行处理失败情况)
发送即忘(fire-and-forget)场景

两句话概括就是:

  • 当你需要可靠的、对协程友好的发送方式时,使用 emit
  • 当你希望实现无阻塞、快速触发且不暂停的操作时,使用 tryEmit

三、重要注意事项

  1. callbackFlow 的特殊情况

    kotlin 复制代码
    callbackFlow {
        callback { data ->
            // 这里必须用 trySend,因为 callback 不是挂起上下文
            trySend(data)
        }
    }
  2. MutableSharedFlow 的配置影响

    kotlin 复制代码
    val flow = MutableSharedFlow<Int>(
        replay = 0,
        extraBufferCapacity = 10  // 缓冲区大小影响 tryEmit 成功率
    )
  3. 性能考虑

    • emit():更安全,保证数据不丢失
    • tryEmit():性能更好,但可能丢失数据

总结一下二者的区别:

特性 emit() tryEmit()
挂起函数
等待收集者 是(如果缓冲区已满)
返回类型 Unit Boolean
是否立即发射? 否(可能会挂起) 是(如果缓冲区允许)
是否可在协程中使用? 必须 不需要

四、最佳实践建议

  • 优先使用 emit() ,除非有特殊性能需求
  • flow {} 构建器中总是使用 emit()
  • 在回调接口中(如 callbackFlow)使用 trySend()/tryEmit()
  • 对于状态更新,如果无法确保在协程上下文中,使用 tryEmit() 但要有失败处理
  • 明确处理 tryEmit() 返回的 false 情况

简单总结:emit() 是安全的阻塞版本,tryEmit() 是非阻塞的尝试版本,可能失败丢数据。

相关推荐
ANYOUZHEN7 小时前
bugku shell
android
南宫码农10 小时前
我的电视 - Android原生电视直播软件 完整使用教程
android·开发语言·windows·电视盒子
道亦无名10 小时前
音频数据特征值提取 方法和步骤
android·音视频
Lancker10 小时前
定制侠 一个国产纯血鸿蒙APP的诞生过程
android·华为·智能手机·鸿蒙·国产操作系统·纯血鸿蒙·华为鸿蒙
2601_9498095912 小时前
flutter_for_openharmony家庭相册app实战+通知设置实现
android·javascript·flutter
液态不合群12 小时前
【面试题】MySQL 中 count(*)、count(1) 和 count(字段名) 有什么区别?
android·数据库·mysql
雪球Snowball14 小时前
【Android关键流程】资源加载
android
2501_9159184114 小时前
常见 iOS 抓包工具的使用,从代理抓包、设备抓包到数据流抓包
android·ios·小程序·https·uni-app·iphone·webview
灯火不休ᝰ15 小时前
[kotlin] 从Java到Kotlin:掌握基础语法差异的跃迁指南
java·kotlin·安卓
墨月白15 小时前
[QT]QProcess的相关使用
android·开发语言·qt