用通俗易懂 + Android 开发实战的方式,详细讲解 Kotlin Flow 中的 retryWhen 操作符

一、一句话理解 retryWhen

retryWhen 是 Flow 的"智能重试"机制:当流抛出异常时,你可以决定"要不要重试"以及"重试几次"。

它就像一个"故障处理员":

  • Flow 执行出错了(比如网络请求失败)
  • retryWhen 问你:"要不要再试一次?"
  • 你根据错误类型、重试次数等条件,返回 true(重试)或 false(放弃)

二、为什么需要它?(对比普通 retry

Kotlin Flow 还有一个简单的 retry(n)

复制代码
flow.retry(3) // 出错就重试最多 3 次,无条件

但现实开发中,你往往需要更精细的控制,比如:

  • 只对网络超时 重试,不对认证失败重试
  • 重试时加延迟(避免雪崩)
  • 最多重试 2 次
  • 用户点击"重试按钮"后再试(手动重试)

👉 这时候就要用 retryWhen


三、基本语法

复制代码
flow.retryWhen { cause: Throwable, attempt: Long ->
    // cause:抛出的异常
    // attempt:当前是第几次重试(从 0 开始)
    // 返回 true:重试
    // 返回 false:不再重试,异常向上抛出
}

⚠️ 注意:retryWhen 的 lambda 是 suspend 函数 ,你可以在里面 delay()


四、Android 实战例子 📱

场景:网络请求失败时智能重试

假设你有一个 Repository 方法:

复制代码
suspend fun fetchUser(): User {
    // 模拟网络请求,可能抛出 IOException(网络问题)或 AuthException(登录过期)
}

在 ViewModel 中用 Flow 包装它:

复制代码
val userFlow = flow {
    emit(fetchUser()) // 可能抛异常
}

现在加上 retryWhen

复制代码
val userWithRetry = userFlow.retryWhen { cause, attempt ->
    val maxRetries = 2

    // 1. 如果重试次数超过上限,不再重试
    if (attempt >= maxRetries) {
        return@retryWhen false
    }

    // 2. 只对特定异常重试(比如网络问题)
    if (cause is IOException || cause is SocketTimeoutException) {
        // 3. 重试前等一会儿(指数退避)
        delay((1000 * (attempt + 1)).toLong()) // 第1次等1秒,第2次等2秒
        return@retryWhen true // 重试!
    }

    // 其他异常(如认证失败)不重试
    false
}

✅ 这样:

  • 网络抖动?自动重试 2 次,每次间隔更长
  • Token 过期?直接报错,让用户重新登录

五、配合 UI:显示"重试按钮"

有时候你不想自动重试,而是等用户点击"重试"按钮再试。

这时候可以把重试逻辑和 UI 事件结合:

复制代码
class UserViewModel : ViewModel() {
    private val _retrySignal = MutableSharedFlow<Unit>()
    
    val uiState = flow {
            emit(fetchUser())
        }
        .retryWhen { cause, _ ->
            // 等待用户点击"重试"
            _retrySignal.first() // 挂起,直到收到信号
            true // 收到信号就重试
        }
        .catch { error ->
            // 如果最终还是失败(比如用户没点重试),转为 Error 状态
            emit(UiState.Error(error))
        }
        .stateIn(viewModelScope, SharingStarted.WhileSubscribed(), UiState.Loading)

    fun onRetryClicked() {
        viewModelScope.launch {
            _retrySignal.emit(Unit) // 发送重试信号
        }
    }
}

UI 中:

复制代码
when (uiState) {
    is UiState.Error -> Button(onClick = { viewModel.onRetryClicked() }) { Text("重试") }
}

✅ 这是 "手动重试" 的经典模式!


六、注意事项 & 常见坑

1. ❌ 不要无限重试!
复制代码
retryWhen { _, _ -> true } // 永远重试 → 死循环!

✅ 一定要加次数限制或条件判断。

2. ⏱️ 重试前加 delay

否则可能瞬间发起大量请求,压垮服务器。

3. 🧪 只对"可恢复的错误"重试
  • ✅ 可重试:网络超时、503 服务不可用
  • ❌ 不可重试:401 未授权、400 参数错误、数据不存在
4. 🔁 retryWhen 会重新执行整个上游 Flow

比如:

复制代码
flow {
    println("开始请求") // 每次重试都会打印!
    emit(api.call())
}.retryWhen { ... }

七、对比其他重试方式

方式 特点 适用场景
retry(n) 简单,无条件重试 n 次 快速原型、简单任务
retryWhen 灵活,可判断异常、加延迟、手动触发 Android 生产环境推荐
手动循环 + try/catch 完全控制 复杂逻辑,但代码啰嗦

八、总结一句话

retryWhen 是 Flow 中实现"智能、可控重试"的核心操作符,特别适合 Android 开发中处理网络请求失败、数据库临时错误等场景,让你的 App 更健壮、用户体验更好。

配合 catchStateFlow 和 UI 重试按钮,你可以构建出非常专业的错误处理流程!


现在使用 Retrofit + Coroutines + Flow 做网络层,retryWhen 几乎是必备技能!

相关推荐
网络精创大傻2 小时前
PHP 与 Node.js:实际性能对比
开发语言·node.js·php
天选之女wow2 小时前
【代码随想录算法训练营——Day60】图论——94.城市间货物运输I、95.城市间货物运输II、96.城市间货物运输III
android·算法·图论
snakecy2 小时前
过关斩将编程题
开发语言·python
diannao7202 小时前
实时将大模型的解决方案转换为随机应变的机器人指令
开发语言·windows·python·机器人
沐怡旸3 小时前
【底层机制】Android对Linux线程调度的移动设备优化深度解析
android·面试
Nebula_g3 小时前
C语言应用实例:斐波那契数列与其其他应用
c语言·开发语言·后端·学习·算法
梅梅绵绵冰3 小时前
SpringAOP的相关概念
java·开发语言
Xiaoyu Wang3 小时前
GC垃圾回收
java·开发语言·jvm
陈佳梁3 小时前
构造器(详解)
java·开发语言