Kotlin - flow

kotlin flow

  • flow (阻塞)
fun flow01(){
        runBlocking {

            val flow = flowOf(5, 4, 3, 2, 1).onEach{
                delay(2000)
            }
            flow.collect {
                println(it)
            }
        }
    }
suspend fun flow02(){

        val flow = flowOf(5, 4, 3, 2, 1).onEach{
            delay(2000)
        }
        flow.collect {
            println(it)
        }
    }
  • measureTimeMillis
suspend fun flow03(){

        val time = measureTimeMillis {

            flow<Any> {
                for (num in 1 until 5) {
                    delay(2000)
                    emit(num)
                }
            }.collect {
                delay(500)
                println(it)
            }
        }
    }
  • 切换线程
suspend fun flow04(){

        val time = measureTimeMillis {

            flow<Any> {
                for (num in 1 until 5) {
                    delay(2000)
                    emit(num)
                }
            }.flowOn(Dispatchers.IO).collect {
                println(it)
            }
        }
    }
  • 取消flow

如果flow是在一个挂起函数内被挂起了,那么flow可以被取消,否则不能取消

suspend fun flow05(){

        val time = measureTimeMillis {

           withTimeoutOrNull(2000){
               flow<Any> {
                   for (num in 1 until 5) {
                       delay(2000)
                       emit(num)
                   }
               }.collect {
                   println(it)
               }
           }
        }
    }
  • 终端操作符 小结
    * collect
    * single/first
    * toList/toSet/toCollection
    * count
    * fold/reduce
    * launchIn/produceIn/broadcastIn
  • flow 生命周期
suspend fun flow06(){

        (1 until 5).asFlow().onEach {
            if (it == 4) throw RuntimeException("Error on $it")
        }.onStart { println("start flow") }
            .onEach { println(it) }
            .catch { println(it.localizedMessage.toString()) }
            .onCompletion { println(" completion") }
            .collect { println(it.toString()) }
    }
  • imperative 通过try...finally实现
suspend fun flow07(){
       runBlocking {
           try {
               flow<Any> {
                   for (num in 1 until 6) {
                       delay(2000)
                       emit(num)
                   }
               }.collect {
                   println(it)
               }
           }
           finally {
               println("done")
           }
       }
    }
  • declarative 通过onCompletion实现
suspend fun flow08(){
        runBlocking {
            flow {
                for (num in 1 until 6) {
                    delay(2000)
                    emit(num)
                }
            }.onCompletion { println("done") }
                .collect {
                    println(it)
                }
        }
    }
  • flow 的缓存策略
    * MISSING 没有指定策略,,不会通过OnNext发射的数据做缓存和丢弃
    *
    * ERROR:如果放入flowable的异步缓存池中的数据超限,就会抛出异常
    *
    * BUFFER:对应即将指定的.buffer(),没有固定大小,可以无限制添加数据,不会抛出异常,但是会oom。
    * 另外:buffer可以并发的执行任务,它是floeOn之外的另一种方式,只是不能显示的指定Dispatcher
    *
    * DROP:异步缓存池满了,就会丢弃即将放入缓存池中的数据。
    *
    * LATEST:对应即将指定的.conflate(),异步缓存池满了,
    * 就会丢弃即将放入缓存池中的数据,不同于DROP的是:LATEST会将最后一条数据强行放入缓存池。
  • flow实现并行 通过flatMapMerge
suspend fun flow09(){
        val elements = arrayListOf<Int>()
        for (num in 0 until 100) {
            elements.add(num)
        }
        elements.asFlow().flatMapMerge {
            flow {
                emit(it)
            }.flowOn(Dispatchers.IO)

        }.collect {
            println(it)
        }
    }
  • 转换操作符 transform

transform ,可以无限制的调用emit,这是它跟map最大的区别,它还可以使用emit发射所有你想要发射的内容

suspend fun flow10(){
        (1 until 5).asFlow().transform {
            emit(it *2)
            delay(2000)
            emit(it * it)
        }
            .collect {
                println(it)
            }
    }
  • 限制发射符 take,只取指定的前几个emit发射的值,其它不管
suspend fun flow11(){
        (1 until 5).asFlow()
            .take(3)
            .collect {
                println(it)
            }
    }
  • reduce 能够对集合进行计算操作
suspend fun flow12(){
        // 计算阶乘
        val sum = (1 until 5).asFlow()
            .map { it * it }
            .reduce { a, b -> a * b }
        println(sum)
    }
  • fold 它有一个initial
suspend fun flow13(){
        // 计算阶乘
        val sum = (1 until 5).asFlow()
            .map { it * it }
            .fold(0){a,b ->
                a * b
            }
    }
  • 合并操作符 zip
suspend fun flow14(){
        val flowA = (0 until 5).asFlow()
        val flowB = flowOf("zero","one","two","three","four")
        flowA.zip(flowB) {a,b ->
            "$a and $b"
        }.collect {
            println(it)
        }
    }
  • combine 这家伙也是合并,flowA每次发出新的元素,会将其与flowB最新的元素合并
suspend fun flow15(){
        val flowA = (0 until 5).asFlow().onEach { delay(500) }
        val flowB = flowOf("zero","one","two","three","four").onEach { delay(1000) }
        flowA.combine(flowB) { a,b ->
            "$a and $b"
        }.collect {
            println(it)
        }
    }
  • flattenConcat 多个流合并的串联执行
suspend fun flow16(){
        val flowA = (0 until 5).asFlow()
        val flowB = flowOf("zero","one","two","three","four")
        flowOf(flowA,flowB)
            .flattenConcat()
            .collect {
                println(it)
            }
    }
  • flatMapConcat
suspend fun flow17(){
        var starting:Long = 0
        (0 until 10).asFlow()
            .onStart { starting = System.currentTimeMillis() }
            .onEach { delay(1000) }
            .flatMapConcat {
                flow {
                    emit("$it first")
                    delay(500)
                    emit("$it second")
                }
            }.collect {
                println("$it at ${System.currentTimeMillis() - starting} -> from start")
            }
    }
  • flatMapLatest 当发射新值之后,上一个flow就会被取消
suspend fun flow18(){
        var starting:Long = 0
        (0 until 10).asFlow()
            .onStart { starting = System.currentTimeMillis() }
            .onEach { delay(1000) }
            .flatMapLatest {
                flow {
                    emit("$it first")
                    delay(500)
                    emit("$it second")
                }
            }.collect {
                println("$it at ${System.currentTimeMillis() - starting} -> from start")
            }
    }
  • channelFlow (异步非阻塞)
suspend fun channelFlow01(){

        val time = measureTimeMillis {

            channelFlow<Any> {
                for (num in 1 until 5) {
                    delay(2000)
                    send(num)
                }
            }.collect {
                delay(500)
                println(it)
            }

        }
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,843评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,538评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,187评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,264评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,289评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,231评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,116评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,945评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,367评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,581评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,754评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,458评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,068评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,692评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,842评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,797评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,654评论 2 354

推荐阅读更多精彩内容