Kotlin Flow 看这一篇 带你入门~

}

}

我们运行main方法,则只有1 2 两个数字进行了打印

1

2

Process finished with exit code 0

Flow的操作符


类似集合的函数是Api,Flow中也有许多操作符,这里我们简单举几个例子:

map

使用map我们可以将最终结果映射为其他类型,代码如下所示:

fun changeData(value: Int): String {

return “打印的结果是:${value}”

}

fun main() {

runBlocking {

loadData1().map {

changeData(it)

}.collect{

println(it)

}

}

}

我们通过map操作符将结果映射为字符串的形式,运行main 打印结果如下所示:

打印的结果是:1

打印的结果是:2

打印的结果是:3

Process finished with exit code 0

filter操作符

通过filter 我们可以对结果集添加过滤条件,如下所示,我们仅打印出大于1的值

runBlocking {

loadData1().filter {

it > 1

}.collect {

println(it)

}

}

故 打印结果如下所示:

2

3

Process finished with exit code 0

所有的操作符都是可以一起使用的,并非只能单独使用。

我们上面调用的collect是末端操作符,在Flow中除了collect之外 还有toList、reduce、fold等操作符。

toList操作符我们可以很明显的知道意为转换为list集合,而reduce 和 fold 则可将最终的值转为单一的值。

fun main() {

runBlocking {

var data = loadData1().reduce { a, b ->

a + b

}

println(data)

}

}

如上代码,我们将Flow的每个结果最终求和,打印结果如下所示:

6

Process finished with exit code 0

flowOn

Flow的代码块是执行在执行时的上下文中,比如 我们不能通过在flow中指定线程来运行Flow代码中的代码,如下所示:

fun loadData1() = flow {

withContext(Dispatchers.Default){

for (i in 1…3) {

delay(1000)

emit(i)

}

}

}

fun main() {

runBlocking {

loadData1().collect { value -> println(“Collected $value”) }

}

}

此种运行方式,将会抛出异常

Exception in thread “main” java.lang.IllegalStateException: Module with the Main dispatcher had failed to initialize. For tests Dispatchers.setMain from kotlinx-coroutines-test module can be used

at kotlinx.coroutines.internal.MissingMainCoroutineDispatcher.missing(MainDispatchers.kt:113)

… …

那么我们如何指定Flow代码块中的上下文呢,我们需要使用flowOn操作符,我们将Flow代码块中的代码指定在IO线程中,代码如下所示:

fun loadData1() = flow {

for (i in 1…3) {

delay(1000)

emit(i)

}

}.flowOn(Dispatchers.IO)

这样我们就把Flow代码块中的事情放到了IO线程中。

buffer操作符

我们在Kotlin 协程 看这一篇就够了 中曾了解过,协程可以提升并发请求的效率,而在Flow代码块中,每当有一个处理结果 我们就可以收到,但如果处理结果也是耗时操作,我们来看下需要多长时间来处理,我们在打印前间隔两秒,并记录开始和完成的时间,代码如下所示:

var startTime: Long = 0

var endTime: Long = 0

fun loadData1() = flow {

startTime = System.currentTimeMillis() / 1000

for (i in 1…3) {

delay(1000)

emit(i)

}

}

fun main() {

runBlocking {

loadData1().collect { value ->

delay(2000)

println(“$value”)

}

endTime = System.currentTimeMillis() / 1000

println(“处理时间:${endTime - startTime}s”)

}

}

运行main方法得到结果如下:

1

2

3

处理时间:9s

Process finished with exit code 0

我们可以看到,处理三个数据,一共使用了9秒钟的时间。

buffer操作符可以使发射和收集的代码并发运行,从而提高效率,我们添加buffer代码如下所示:

fun main() {

runBlocking {

loadData1().buffer().collect { value ->

delay(2000)

println(“$value”)

}

endTime = System.currentTimeMillis() / 1000

println(“处理时间:${endTime - startTime}s”)

}

}

再次运行main方法,结果如下所示:

1

2

3

处理时间:8s

Process finished with exit code 0

由此看出,时间较少了接近1s(/1000为近似值),不要小看这小小的1秒,运行在手机上还是相当重要的~

zip操作符

zip操作符,可以合并两个flow,代码如下所示:

fun loadData1() = flow {

for (i in 1…3) {

delay(1000)

emit(“我是j:${i}”)

}

}

fun loadData2() = flow {

for (i in 1…3) {

delay(1000)

emit(“我是i:${i}”)

}

}

fun main() {

runBlocking {

loadData2().zip(loadData1()){

a,b -> “ a , a, a,b”

}.collect{

println(it)

}

}

}

运行结果如下所示:

最后

文章所有资料全部已经打包整理好,另外小编手头上整理了大量Android架构师全套学习资料,Android核心高级技术PDF文档+全套高级学习资料+视频+2021 BAT 大厂面试真题解析

资料展示:

image

image

image

image
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
nBlocking {

loadData2().zip(loadData1()){

a,b -> “ a , a, a,b”

}.collect{

println(it)

}

}

}

运行结果如下所示:

最后

文章所有资料全部已经打包整理好,另外小编手头上整理了大量Android架构师全套学习资料,Android核心高级技术PDF文档+全套高级学习资料+视频+2021 BAT 大厂面试真题解析

资料展示:

[外链图片转存中…(img-zMry5nZG-1715573447071)]

[外链图片转存中…(img-aFj305wM-1715573447074)]

[外链图片转存中…(img-cwPZjVLr-1715573447076)]

[外链图片转存中…(img-5S87WPQI-1715573447077)]
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 28
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值