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/无满足条件元素时抛出异常。
相关推荐
Highcharts.js1 天前
倒置百分比堆叠面积图表示列详解|Highcharts大气成分图表代码
开发语言·信息可视化·highcharts·图表开发·面积图·图表示例·推叠图
2601_958492551 天前
Optimizing Engagement with Freehead Skate - HTML5 Game - Construct 3
前端·html·html5
csdn_aspnet1 天前
C语言 Lomuto分区算法(Lomuto Partition Algorithm)
c语言·开发语言·算法
晨曦中的暮雨1 天前
4.15腾讯 CSIG云服务产线 一面
java·开发语言
存在morning1 天前
【GO语言开发实践】二 GO 并发快速上手
大数据·开发语言·golang
茉莉玫瑰花茶1 天前
工作流的常见模式 [ 1 ]
java·服务器·前端
zhangxingchao1 天前
AI应用开发六:企业知识库
前端·人工智能·后端
山峰哥1 天前
SQL慢查询调优实战:从全表扫描到索引覆盖的完整复盘
前端·数据库·sql·性能优化
xiaoerbuyu12331 天前
开源Java 邮箱 基于SpringBoot+Vue前后端分离的电子邮件
java·开发语言