Android使用协程(Coroutine)优雅的处理多个接口同时请求(网络并发请求)

小鱼儿 2024-03-23 11:52 203阅读 0赞

Android中同时请求多个网络接口

我们在开发过程中,很可能会遇到这样的需求,一个页面中需要同时请求好几个接口。

我在之前写过一篇关于 高德逆地理编码接口返回数据格式不统一以及百度逆地理编码接口返回数据解析失败的踩坑记录 这篇文章,我当时的需求是要结合高德和百度两个接口,合并成一个我想要的数据。在文章中,我们使用了Rxjava的zip操作符来处理的两个接口同时请求的逻辑。大概代码如下:

  1. Observable.zip(
  2. editUnitInfoModel.getBaiduRegeo(location),
  3. editUnitInfoModel.getGaodeRegeo(location),
  4. BiFunction<BaiduRegeoBean, GaodeRegeoBean, LocationDetailsBean> { baidu, gaode ->
  5. val locationDetailsBean = LocationDetailsBean()
  6. if (baidu.status == 0) {
  7. val baiduAddressComponent = baidu.result.addressComponent
  8. locationDetailsBean.adcode = baiduAddressComponent.adcode
  9. locationDetailsBean.city = baiduAddressComponent.city
  10. locationDetailsBean.country = baiduAddressComponent.country
  11. locationDetailsBean.district = baiduAddressComponent.district
  12. locationDetailsBean.province = baiduAddressComponent.province
  13. locationDetailsBean.street = baiduAddressComponent.street
  14. locationDetailsBean.streetNumber = baiduAddressComponent.streetNumber
  15. locationDetailsBean.town = baiduAddressComponent.town
  16. }
  17. if (gaode.status == "1") {
  18. locationDetailsBean.townCode = gaode.regeocode.addressComponent.towncode
  19. }
  20. LogUtils.i("合并后的定位信息为:${locationDetailsBean}")
  21. return@BiFunction locationDetailsBean
  22. }
  23. )

具体实现逻辑感兴趣的可以去看看那篇文章。本篇文章中,我们来使用Kotlin协程来实现这个需求,你会发现,使用协程来处理,代码非常的简洁。


Kotlin协程处理并发

如果你对协程还不了解,建议去官网先学习一下。

这里我们使用两个方法来模拟网络请求

  1. suspend fun one(): Int {
  2. delay(1500)
  3. return 1
  4. }
  5. suspend fun two(): Int {
  6. delay(1500)
  7. return 2
  8. }

两个方法都是可挂起的方法,分别返回1和2

假如这两个接口之间没有联系,我们想让他们并发执行的话,我们可以使用async和await配合使用,代码如下

  1. fun main() {
  2. GlobalScope.launch {
  3. /*measureTimeMillis返回给定的block代码的执行时间*/
  4. val time = measureTimeMillis {
  5. val sum = withContext(Dispatchers.IO) {
  6. val one = async { one() }
  7. val two = async { two() }
  8. one.await() + two.await()
  9. }
  10. println("两个方法返回值的和:${sum}")
  11. }
  12. println("执行耗时:${time}")
  13. }
  14. println("----------------")
  15. /*应为上面的协程代码并不会阻塞掉线程,所以我们这里让线程睡4秒,保证线程的存活,在实际的Android开发中无需这么做*/
  16. Thread.sleep(4000)
  17. }

下面我们来运行看看执行结果:

在这里插入图片描述

可以看到,我们顺利的拿到了想要的结果。而且对比使用RxJava的实现方法,会发现,协程的代码会少很多,且结构更加的清晰。

Kotlin协程处理多个耗时操作按顺序执行

假如你有这样的需求,接口B的参数是接口A返回的结果,那这样的话,使用协程依然能够很简洁的实现。
我们把两个方法稍微改动一下

  1. suspend fun one(): Int {
  2. delay(1000)
  3. return 1
  4. }
  5. suspend fun two(int: Int): Int {
  6. delay(2000)
  7. return 2 + int
  8. }

方法two接收一个参数,该参数是由方法one返回的结果决定的

实现代码如下

  1. fun main() {
  2. GlobalScope.launch {
  3. /*measureTimeMillis返回给定的block代码的执行时间*/
  4. val time = measureTimeMillis {
  5. val sum = withContext(Dispatchers.IO) {
  6. val one = one()
  7. val two = two(one)
  8. one + two
  9. }
  10. println("两个方法返回值的和:${sum}")
  11. }
  12. println("执行耗时:${time}")
  13. }
  14. println("----------------")
  15. /*应为上面的协程代码并不会阻塞掉线程,所以我们这里让线程睡4秒,保证线程的存活,在实际的Android开发中无需这么做*/
  16. Thread.sleep(4000)
  17. }

在这里插入图片描述

可以看到我们得到了想要的结果。

kotlin协程总结

通过这两个小例子可以发现,无论你是要实现并发还是要指定顺序的执行多个异步方法,协程都能很简洁的实现。
在实际的Android开发中,我们可以结合Retrofit2.6.0及以后的版本更加优雅的实现网络请求等异步逻辑。
我个人认为协程完全可以替代RxJava去实现异步逻辑,且代码更加简洁。

关于结合Retrofit的使用请看:Retrofit+kotlin Coroutines(协程)+mvvm(Jetpack结构组件)实现更简洁的网络请求

好了,本篇文章就是这样

发表评论

表情:
评论列表 (有 0 条评论,203人围观)

还没有评论,来说两句吧...

相关阅读