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() 是非阻塞的尝试版本,可能失败丢数据。

相关推荐
2501_915106322 小时前
iOS 上架费用解析,哪些成本可以通过流程优化降低。
android·ios·小程序·https·uni-app·iphone·webview
顾林海3 小时前
Android暗黑模式适配全攻略:从入门到精通,告别"阴间配色"
android·面试·性能优化
唔663 小时前
出厂前一次性授权
android
勤劳打代码3 小时前
水到渠成 —— 从项目出发的 Claude SKILL 实践
ai编程·claude·android jetpack
崇山峻岭之间3 小时前
Matlab学习记录12
android·学习·matlab
jllllyuz4 小时前
C# 面向对象图书管理系统
android·开发语言·c#
灵感菇_4 小时前
Android图片加载框架 Glide全面解析
android·缓存·glide
、BeYourself4 小时前
Android 开发:交错网格、卡片视图与 RecyclerView 实战
android·android-studio
apihz4 小时前
免费手机号归属地查询API接口详细教程
android·java·运维·服务器·开发语言