kotlin Flow first() last()总结

一、核心函数定义

以下是 first()/firstOrNull()/last()/lastOrNull() 的核心逻辑,其中 last 系列以「Flow 的 block 执行完毕」为核心判断依据:

函数 核心行为(无 predicate) 带 predicate 时的行为 异常/边界处理
first() 触发 Flow 收集,拿到发射的第一个元素后立即取消收集(无需等待 block 执行完毕) 拿到第一个满足条件的元素后立即取消收集 空 Flow/无满足条件的元素 → 抛出 NoSuchElementException
firstOrNull() first() 逻辑完全一致 first() 逻辑完全一致 空 Flow/无满足条件的元素 → 返回 null(不抛出异常)
last() 触发 Flow 收集,等待 Flow 的整个 block 执行完毕后,取发射的最后一个元素 等待 Flow 的 block 执行完毕后,取最后一个满足条件的元素 空 Flow/无满足条件的元素 → 抛出 NoSuchElementException;block 无限执行 → 永久挂起
lastOrNull() last() 逻辑完全一致 last() 逻辑完全一致 空 Flow/无满足条件的元素 → 返回 null;block 无限执行 → 永久挂起

关键说明

  • 所有函数均为 Flow 的终端操作符,调用后会立即触发 Flow 收集;
  • predicate 仅改变"筛选元素的规则",不改变"取第一个/最后一个""是否等待 block 执行完毕"的核心逻辑。

二、不同 Flow 类型下的行为差异

1. 普通 Flow(冷流)

普通 Flow 由 flow { ... } 定义,此处的 { ... } 即为该 Flow 的 block,每次调用终端操作都会重新执行该 block:

函数 block 可执行完毕的有限 Flow(如 flow { emit(1); emit(2); emit(3) } block 无限执行的无限 Flow(如 flow { while(true) { emit(1) } }
first()/firstOrNull() 正常拿到第一个(满足 predicate)元素,拿到后立即取消 block 执行 正常拿到第一个(满足 predicate)元素,拿到后立即取消 block 执行
last()/lastOrNull() 等待 block 完全执行完毕,正常返回最后一个(满足 predicate)元素 block 无限执行,函数永久挂起,无返回结果

示例验证

kotlin 复制代码
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.flow
import kotlinx.coroutines.runBlocking

fun main() = runBlocking {
    // 有限普通 Flow(block 可执行完毕)
    val finiteFlow = flow {
        emit(1)
        delay(10)
        emit(2)
        delay(100)
        emit(3) // block 执行完毕
    }
    println(finiteFlow.first())  // 输出 1(立即返回,取消 block 执行)
    println(finiteFlow.last())   // 输出 3(等待 block 执行完毕后返回)

    // 无限普通 Flow(block 无限执行)
    val infiniteFlow = flow {
        var i = 0
        while (true) { // 无限循环,block 永不结束
            emit(i++)
            delay(100)
        }
    }
    println(infiniteFlow.first()) // 输出 0(立即返回)
    // println(infiniteFlow.last()) // 永久挂起,无输出
}
2. StateFlow/SharedFlow(热流)

StateFlow/SharedFlow 无用户自定义的 flow { ... } block,其内部维护了永久活跃的发射逻辑(可理解为内部 block 无限执行),且为热流(创建后持续活跃,无需每次收集重新执行逻辑):

函数 StateFlow(如 MutableStateFlow(0) SharedFlow(如 MutableSharedFlow<Int>()
first()/firstOrNull() 立即拿到当前最新值(StateFlow 初始值/已更新值),拿到后取消收集 无缓存值时等待第一个值发射,拿到后取消收集;有缓存值时直接拿第一个缓存值
last()/lastOrNull() 内部 block 无限执行,函数永久挂起;手动取消协程则抛出 CancellationException 内部 block 无限执行,函数永久挂起;手动取消协程则抛出 CancellationException

示例验证

kotlin 复制代码
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking

fun main() = runBlocking {
    // StateFlow 示例
    val stateFlow = MutableStateFlow(0)
    println(stateFlow.first()) // 输出 0(立即拿到初始值,取消收集)
    stateFlow.value = 1
    // println(stateFlow.last()) // 永久挂起,无输出

    // SharedFlow 示例(无初始缓存)
    val sharedFlow = MutableSharedFlow<Int>()
    launch { delay(100); sharedFlow.emit(1) }
    println(sharedFlow.first()) // 等待 100ms 输出 1(拿到后取消收集)
    // println(sharedFlow.last()) // 永久挂起,无输出
}

总结

  1. first()/firstOrNull() 核心是「拿第一个元素即终止收集」,无需等待 block 执行完毕,对所有类型 Flow 均能正常返回(SharedFlow 无缓存时仅需等待第一个值发射);
  2. last()/lastOrNull() 核心是「等 Flow 的 block 执行完毕再拿最后一个元素」,仅适用于 block 可执行完毕的有限普通 Flow,对无限普通 Flow/StateFlow/SharedFlow 会永久挂起;
  3. 实际开发中优先使用 firstOrNull()/lastOrNull(),避免空 Flow/无满足条件元素时抛出异常。
相关推荐
W.D.小糊涂1 小时前
gpu服务器安装windows+ubuntu24.04双系统
c语言·开发语言·数据库
用头发抵命2 小时前
Vue 3 中优雅地集成 Video.js 播放器:从组件封装到功能定制
开发语言·javascript·ecmascript
似水明俊德2 小时前
02-C#.Net-反射-学习笔记
开发语言·笔记·学习·c#·.net
蓝冰凌2 小时前
Vue 3 中 defineExpose 的行为【defineExpose暴露ref变量】详解:自动解包、响应性与实际使用
前端·javascript·vue.js
奔跑的呱呱牛2 小时前
generate-route-vue基于文件系统的 Vue Router 动态路由生成工具
前端·javascript·vue.js
于先生吖2 小时前
Java框架开发短剧漫剧系统:后台管理与接口开发
java·开发语言
柳杉3 小时前
从动漫水面到赛博飞船:这位开发者的Three.js作品太惊艳了
前端·javascript·数据可视化
khddvbe3 小时前
C++并发编程中的死锁避免
开发语言·c++·算法