线程切换哪家强?RxJava与Flow的操作符对比

上面蓝色绿色部分因为observeOn的存在分别切换到了不同线程执行

just

RxJava的初学者经常会犯的一个错误是在Observable.just(...)里做耗时任务。 just并不是接受lambda,所以是立即执行的,不受subscribeOn的影响

如上,loadDataSync()不会在io执行,

想要在io执行,需要使用Observable.deffer{}

flatMap

结合上面介绍的RxJava的线程切换,看下面这段代码

如果我们希望loadData(id)并发执行,那么上面的写法是错误的。

subscribe(io())意味着其上游的数据在单一线程中串行发射。因此虽然flatMap{}返回多个Observable, 都是都在单一线程中订阅,多个loadData始终运行在同一线程。

代码经过一下修改后,可以达到并发执行的效果:

当订阅flatMap返回的Observable时,通过subscribeOn分别指定订阅线程。

其他类似flatMap这种涉及多个Observable订阅的操作符(例如mergezip等),需要留意各自的subscribeOn的线程,以防不符合预期的行为出现。

2. Flow

===================================================================


接下来看一下 Flow的线程切换 。

Flow是基于CoroutineContext进行线程切换,所以这部分内容需要你对Croutine事先有基本的了解。

flowOn类似于RxJava的subscribeOn,Flow中没有对应observeOn的操作符,因为collect是一个suspend函数,必须在CoroutineScope中执行,所以响应线程是由CoroutineContext决定的。例如你在main中执行collect,那么响应线程就是Dispatcher.Main

flowOn

flowOn类似于subscribeOn,因为它们都可以用来决定上游线程 在这里插入图片描述

上面代码中,flowOn前面代码将会在IO执行。

subscribeOn不同的是,flowOn允许出现多次,每个都会影响其前面的操作 在这里插入图片描述

上面代码,根据颜色可以看出来flowOn影响的范围

launchIn

collect是suspend函数,所以后续代码因为协程挂起不会继续执行 在这里插入图片描述

所以上面代码可能会不符合预期,因为第一个collect不走完第二个走不到。

正确的写法是为每个collect单独起一个协程 [图片上传中…(image-d7a185-1616592924551-13)]

或者使用launchIn,写法更加优雅 在这里插入图片描述

launchIn不会挂起协程,所以与RxJava的subscribe更加接近。

通过名字可以感觉出来launchIn只不过是之前例子中launch的一个链式调用的语法糖。

flowOf

flowOf类似于Observable.just(),需要注意flowOf内的内容是立即执行的,不受flowOn影响 在这里插入图片描述

希望calculate()运行在IO,可以使用flow{ }

flatMapMerge

flatMapMerge类似RxJava的flatMap 在这里插入图片描述

如上,2个item各自flatMap成2个item,即一共发射了4条数据,日志输出如下:

inner: pool-2-thread-2 @coroutine#4

inner: pool-2-thread-3 @coroutine#5

inner: pool-2-thread-3 @coroutine#5

inner: pool-2-thread-2 @coroutine#4

collect: pool-1-thread-2 @coroutine#2

collect: pool-1-thread-2 @coroutine#2

collect: pool-1-thread-2 @coroutine#2

collect: pool-1-thread-2 @coroutine#2

复制代码

通过日志我们发现flowOn虽然写在flatMapMerge外面,inner的日志却可以打印在多个线程上(都来自pool2线程池),这与flatMap是不同的,同样场景下flatMap只能运行在线程池的固定线程上。

如果将flowOn写在flatMapMerge内部

结果如下:

inner: pool-2-thread-2 @coroutine#6

inner: pool-2-thread-1 @coroutine#7

inner: pool-2-thread-2 @coroutine#6

inner: pool-2-thread-1 @coroutine#7

collect: pool-1-thread-3 @coroutine#2

collect: pool-1-thread-3 @coroutine#2

collect: pool-1-thread-3 @coroutine#2

collect: pool-1-thread-3 @coroutine#2

复制代码

inner仍然打印在多个线程,flowOn无论写在flatMapMerge内部还是外部,对flatMapMerge内的处理没有区别。

但是flatMapMerge之外还是有区别的,看下面两段代码 在这里插入图片描述

通过颜色可以知道flowOn影响的范围,向上追溯到flowOf为止

3. Summary

======================================================================


RxJava的Observable与Coroutine的Flow都支持线程切换,相关API的对比如下:

| | 线程池调度 | 线程操作符 | 数据源同步创建 | 异步创建 | 并发执行 |

| :-- | :-- | :-- | :-- | :-- | :-- |

| RxJava | Schedulers (io(), computation(), mainThread()) | subscribeOn, observeOn | just | deffer{} | flatMap(inner subscribeOn) |

| Flow | Dispatchers (IO, Default, Main) | flowOn | flowOf | flow{} | flatMapMerge(inner or outer flowOn) |

最后通过一个例子看一下如何将代码从RxJava迁移到Flow

RxJava

RxJava代码如下:

使用到的Schedulers定义如下: 在这里插入图片描述

代码执行结果:

1: pool-1-thread-1

1: pool-1-thread-1

1: pool-1-thread-1

2: pool-3-thread-1

2: pool-3-thread-1

2: pool-3-thread-1

inner 1: pool-4-thread-1

inner 1: pool-4-thread-2

inner 1: pool-4-thread-1

inner 1: pool-4-thread-1

inner 1: pool-4-thread-2

inner 1: pool-4-thread-2

inner 1: pool-4-thread-3

inner 2: pool-5-thread-1

inner 2: pool-5-thread-2

3: pool-5-thread-1

inner 2: pool-5-thread-2

inner 1: pool-4-thread-3

inner 2: pool-5-thread-2

inner 2: pool-5-thread-3

3: pool-5-thread-1

3: pool-5-thread-1

3: pool-5-thread-1

end: pool-6-thread-1

end: pool-6-thread-1

inner 1: pool-4-thread-3

end: pool-6-thread-1

3: pool-5-thread-1

inner 2: pool-5-thread-1

3: pool-5-thread-1

inner 2: pool-5-thread-3

inner 2: pool-5-thread-1

end: pool-6-thread-1

3: pool-5-thread-3

3: pool-5-thread-3

end: pool-6-thread-1

inner 2: pool-5-thread-3

3: pool-5-thread-3

end: pool-6-thread-1

end: pool-6-thread-1

end: pool-6-thread-1

end: pool-6-thread-1

复制代码

代码较长,通过颜色标记法帮我们理清线程关系

上色后一目了然了,需要特别注意的是由于flatMap中切换了数据源的同时切换了线程,所以打印 3的线程不是s2 而是 s4

Flow

首相创建对应的Dispatcher

然后将代码换成Flow的写法,主要遵循下列原则

  • RxJava通过observeOn切换后续代码的线程

  • Flow通过flowOn切换前置代码的线程

打印结果如下:

1: pool-1-thread-1 @coroutine#6

1: pool-1-thread-1 @coroutine#6

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

总结

可以看出,笔者的工作学习模式便是由以下 「六个要点」 组成:

❝ 多层次的工作/学习计划 + 番茄工作法 + 定额工作法 + 批处理 + 多任务并行 + 图层工作法❞

希望大家能将这些要点融入自己的工作学习当中,我相信一定会工作与学习地更富有成效。

下面是我学习用到的一些书籍学习导图,以及系统的学习资料。每一个知识点,都有对应的导图,学习的资料,视频,面试题目。

**如:我需要学习 **Flutter的知识。(大家可以参考我的学习方法)

  • Flutter 的思维导图(无论学习什么,有学习路线都会事半功倍)

  • Flutter进阶学习全套手册

  • Flutter进阶学习全套视频

大概就上面这几个步骤,这样学习不仅高效,而且能系统的学习新的知识。

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

作学习当中,我相信一定会工作与学习地更富有成效。

下面是我学习用到的一些书籍学习导图,以及系统的学习资料。每一个知识点,都有对应的导图,学习的资料,视频,面试题目。

**如:我需要学习 **Flutter的知识。(大家可以参考我的学习方法)

  • Flutter 的思维导图(无论学习什么,有学习路线都会事半功倍)

[外链图片转存中…(img-xIIpymO1-1712024292673)]

  • Flutter进阶学习全套手册

[外链图片转存中…(img-DKxsHOzd-1712024292673)]

  • Flutter进阶学习全套视频

[外链图片转存中…(img-f3Ek1aJX-1712024292673)]

大概就上面这几个步骤,这样学习不仅高效,而且能系统的学习新的知识。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值