近日 Retrofit 更新到了 2.6.0 版本,内置了对 Kotlin Coroutines 的支持,进一步简化了使用 Retrofit 和协程来进行网络请求的过程。其实纵观编程语言的发展历史,从汇编到 C/C++,从 Java,OC 到 Swift,Kotlin,甚至被纳入教材的 Python,都有一个共同的特点。随着 CPU 性能的越来越强悍,提高生产力似乎都成了现代高级编程语言的共同目标。Kotlin 就是一个好例子,做同样的事情,完成同样的功能,Java 的确需要更多的代码,Kotlin 也的确给 Android 开发提升了效率。特别是在异步任务方面,Kotlin 提供了协程,而这是 Java 所不具备的。关于 Kotlin Coroutines 的介绍,可以阅读我之前的三篇译文:
在 Android 上使用协程(一):Getting The Background
在 Android 上使用协程(二):Getting started
在 Android 上使用协程(三) :Real Work
回到正题,本篇主要介绍 Retrofit 2.6.0 版本中协程的使用方式,不会过多涉及原理。我以我自己的 wanandroid 应用为例进行改造,源代码中 Retrofit 版本是 2.4.0
。这个 wanandroid 是基于 Kotlin + 协程 + LiveData + MVVM
实现的,具体架构可见我的文章 真香!Kotlin+MVVM+LiveData+协程 打造 Wanandroid! ,个人觉得代码还是比较清晰的,很适合作为 Kotlin 的 入门项目。
老版本 Retrofit 的使用
在介绍如何使用 Retrofit 2.6.0
之前,我们先来看一下老版本的 Retrofit 是如何基于 Kotlin Coroutines 工作的,以登录接口为例。
首先在 WanService 接口中作如下定义:
@POST("/user/login")
fun login(@Field("username") userName: String,
@Field("password") passWord: String): Deferred>
复制代码
注意这里使用的返回值是 Deferred
对象,这就意味着使用的时候要通过 await
来获取返回值。那么如何让 Retrofit 直接返回 Deferred
呢?使用的也是 JakeWharton 的开源库:
implementation 'com.jakewharton.retrofit:retrofit2-kotlin-coroutines-adapter:0.9.2'
复制代码
在构建 Client 的时候添加上这个适配器:
...
.addCallAdapterFactory(CoroutineCallAdapterFactory.invoke())
...
复制代码
然后给 LoginRepository 提供一个 suspend 方法:
suspend fun login(userName: String, passWord: String): WanResponse {
return apiCall { WanRetrofitClient.service.login(userName, passWord).await() }
}
复制代码
这里使用 await
来获取 Deferred
的返回值。
最后在 LoginViewModel 中是这样调用的:
fun login(userName: String, passWord: String) {
launch {
val response = withContext(Dispatchers.IO) { repository.login(userName, passWord) }
executeResponse(response, { mLoginUser.value = response.data }, { errMsg.value = response.errorMsg })
}
}
复制代码
launch()
方法做了简单的封装,感兴趣的同学可以到源码中看一下。
以上就是在 Retrofit 2.4.0
中使用协程的基本方式了,其实代码也很简洁。而 Retrofit 2.6.0
让这一切更加简单!就让我们一睹为快吧!
Retrofit 2.6.0 中协程的使用
Talking is cheap, show me the code !
还是上面的登录接口,基于 Retrofit 2.6.0
来改造一下。
第一步,修改 Retrofit 依赖。
implementation 'com.squareup.retrofit2:retrofit:2.6.0'
复制代码
第二步,修改 WanService
中接口的定义。
@POST("/user/login")
suspend fun login(@Field("username") userName: String,
@Field("password") passWord: String): WanResponse
复制代码
看到区别了吗?首先,不再返回 Deferred
对象,而是直接返回我们需要的 WanResponse
对象。其次,使用了 suspend
来修改方法,标记这是挂起函数。
第三步,修改 LoginRepository
中方法定义。
suspend fun login(userName: String, passWord: String): WanResponse {
return apiCall { WanRetrofitClient.service.login(userName, passWord) }
}
复制代码
与之前的版本相比,这里不需要调用 await
方法了。其实并不是不调用了,而是 Retrofit 帮助我们自动调用了。
最后别忘了去除之前添加的 kotlin-coroutines-adapter
,因为我们不再需要人工返回 Deferred
对象,也不再需要手动调用 await
了。
...
//.addCallAdapterFactory(CoroutineCallAdapterFactory.invoke())
...
复制代码
至此,基于 Retrofit 2.6.0
版本的改造就已经完成了。我的 Wanandroid 项目已经完成全部修改,具体修改内容可见 commit。
总结
随着 Kotlin 成为 Android 开发的首选语言,越来越多的新特性都将在 Kotlin 上优先实现。协程作为 Kotlin 的异步利器,很值得我们学习。如果你还没有入手,那么,从我的 Wanandroid 开始吧 !
文章首发微信公众号:
秉心说
, 专注 Java 、 Android 原创知识分享,LeetCode 题解。更多相关知识,扫码关注我吧!