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/无满足条件元素时抛出异常。
相关推荐
Wenweno0o1 天前
0基础Go语言Eino框架智能体实战-chatModel
开发语言·后端·golang
于慨1 天前
Lambda 表达式、方法引用(Method Reference)语法
java·前端·servlet
石小石Orz1 天前
油猴脚本实现生产环境加载本地qiankun子应用
前端·架构
从前慢丶1 天前
前端交互规范(Web 端)
前端
CHU7290351 天前
便捷约玩,沉浸推理:线上剧本杀APP功能版块设计详解
前端·小程序
chenjingming6661 天前
jmeter线程组设置以及串行和并行设置
java·开发语言·jmeter
GISer_Jing1 天前
Page-agent MCP结构
前端·人工智能
王霸天1 天前
💥别再抄网上的Scale缩放代码了!50行源码教你写一个永不翻车的大屏适配
前端·vue.js·数据可视化
小领航1 天前
用 Three.js + Vue 3 打造炫酷的 3D 行政地图可视化组件
前端·github
@大迁世界1 天前
2026年React大洗牌:React Hooks 将迎来重大升级
前端·javascript·react.js·前端框架·ecmascript