fun main() {
runBlocking {
withTimeoutOrNull(2500){
loadData1().collect {
println(it)
}
}
}
}
我们运行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
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
如何做好面试突击,规划学习方向?
面试题集可以帮助你查漏补缺,有方向有针对性的学习,为之后进大厂做准备。但是如果你仅仅是看一遍,而不去学习和深究。那么这份面试题对你的帮助会很有限。最终还是要靠资深技术水平说话。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。建议先制定学习计划,根据学习计划把知识点关联起来,形成一个系统化的知识体系。
学习方向很容易规划,但是如果只通过碎片化的学习,对自己的提升是很慢的。
同时我还搜集整理2020年字节跳动,以及腾讯,阿里,华为,小米等公司的面试题,把面试的要求和技术点梳理成一份大而全的“ Android架构师”面试 Xmind(实际上比预期多花了不少精力),包含知识脉络 + 分支细节。
在搭建这些技术框架的时候,还整理了系统的高级进阶教程,会比自己碎片化学习效果强太多。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
自己的提升是很慢的。
同时我还搜集整理2020年字节跳动,以及腾讯,阿里,华为,小米等公司的面试题,把面试的要求和技术点梳理成一份大而全的“ Android架构师”面试 Xmind(实际上比预期多花了不少精力),包含知识脉络 + 分支细节。
[外链图片转存中…(img-cWmZCoOy-1712797402754)]
在搭建这些技术框架的时候,还整理了系统的高级进阶教程,会比自己碎片化学习效果强太多。
[外链图片转存中…(img-yl1S5ag7-1712797402755)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-sjRbNdnY-1712797402755)]