Kotlin协程-ShareFlow

前言

replay粘性事件的个数,不同消费者订阅后,重新接收replay个之前的数据。

kotlin 复制代码
    fun main() {
        runBlocking {
            val flow = MutableSharedFlow<String>(replay = 1)
            // 第一个订阅
            launch {
                flow.collect{
                    println("collect1:$it")
                }
            }
            // 生产数据
            launch {
                (1..4).forEach{
                    println("emit:$it")
                    flow.emit("$it")
                }
            }
            //第二个订阅者 模拟在生产数据后才订阅
            launch {
                delay(1000)
                flow.collect{
                    println("collect2:$it")
                }
            }
        }
    }
}
输出:
emit:1
emit:2
collect1:1
collect1:2
emit:3
emit:4
collect1:3
collect1:4
collect2:4

replay=n时,多次订阅会将生产者的最后n次事件冲i性能发送一遍。

extraBufferCapacity

额外的缓存池。总的缓冲池extraBufferCapacity+replay个,用于生产速率>消费速率的情况。

kotlin 复制代码
fun main() {
    runBlocking {
        val flow = MutableSharedFlow<String>(replay = 2, extraBufferCapacity = 1)
        // 第一个订阅
        launch {
            flow.collect{
                delay(2000)
                println("collect1:$it")
            }
        }
        // 生产数据
        launch {
            (1..10).forEach{
                flow.emit("$it")
                println("emit:$it")
                delay(100)
            }
        }
    }
}
输出:
emit:1
emit:2
emit:3
emit:4
collect1:1
emit:5
collect1:2
emit:6
collect1:3
emit:7
collect1:4
emit:8
collect1:5
emit:9
collect1:6
emit:10
collect1:7
collect1:8
collect1:9
collect1:10

当缓存池满后,消费掉一个才会生产一个,生产者挂起。extraBufferCapacity=1时,发送和接收交替打印。当extraBufferCapacity>=7时,发送日志打印完毕后才开始打印接收日志。

onBufferOverFlow

缓冲策略,和Java线程池类似。线程池的拒绝策略。

kotlin 复制代码
public enum class BufferOverflow {
    //默认的,当生产速率大于消费速率并且缓冲池已满的情况,会挂起生产者,等待消费者
    SUSPEND,
    //丢弃老的数据
    DROP_OLDEST,
    //丢弃新的数据
    DROP_LATEST
}

StateFlow

StateFlow是一种特殊的ShareFlow。replay=1。必须要有初始值,不能为空,两次数据一致的情况下不会触发第二次。 LiveData相同点: 1、允许多个消费者订阅 2、粘性事件,事件数量=1 3、产生数据太快时都会丢失数据

StateFlow模式是BufferOverflow.DROP_OLDEST。Sharedflow是BufferOverflow.SUSPEND

不同点:

  1. StateFlow必须有默认值
  2. 默认是防抖,LiveData不防抖
  3. StateFlow没有和生命周期绑定

LiveData防抖可使用distinctUntilChanged。StateFlow没有生命周期绑定,可以利用Liecycle,在协程作用域内使用repeatOnLifecycle.

总结

  1. StateFlow是特殊的ShareFlow,replay=1 onBufferOverflow=BufferOverFlow.SUSPEND
  2. StateFlow可以替代LiveData使用
  3. StateFlow必须有默认值,默认防抖,有粘性事件,没有生命周期绑定
  4. ShareFlow没有默认值,粘性事件,可自定义事件数量,有缓冲池
  5. StateFlow表示状态,如View的显示和隐藏,按钮selectd都需要默认值
  6. ShareFlow是存储数据,例如列表list<T>数据,屏幕发生旋转,在Fragment的view消耗重建后的数据恢复
  7. 数据列表用ShareFlow<List<data>>,点赞或收藏数据的变化用StateFlow
相关推荐
fatiaozhang95276 分钟前
中国移动中兴云电脑W132D-RK3528-2+32G_安卓9_ADB开启线刷包
android·adb·电脑·电视盒子·刷机固件·机顶盒刷机·中兴云电脑w132d
selt7919 小时前
Redisson之RedissonLock源码完全解析
android·java·javascript
Yao_YongChao10 小时前
Android MVI处理副作用(Side Effect)
android·mvi·mvi副作用
非凡ghost11 小时前
JRiver Media Center(媒体管理软件)
android·学习·智能手机·媒体·软件需求
席卷全城11 小时前
Android 推箱子实现(引流文章)
android
齊家治國平天下11 小时前
Android 14 系统中 Tombstone 深度分析与解决指南
android·crash·系统服务·tombstone·android 14
maycho12313 小时前
MATLAB环境下基于双向长短时记忆网络的时间序列预测探索
android
思成不止于此14 小时前
【MySQL 零基础入门】MySQL 函数精讲(二):日期函数与流程控制函数篇
android·数据库·笔记·sql·学习·mysql
brave_zhao14 小时前
达梦数据库(DM8)支持全文索引功能,但并不直接兼容 MySQL 的 FULLTEXT 索引语法
android·adb
sheji341614 小时前
【开题答辩全过程】以 基于Android的网上订餐系统为例,包含答辩的问题和答案
android