Retrofit 2.6.0 ! 更快捷的协程体验 !
近日 Retrofit 更新到了 2.6.0 版本,内置了对 Kotlin Coroutines 的支持,进一步简化了使用 Retrofit 和协程来进行网络请求的过程。其实纵观编程语言的发展历史,从汇编到 C/C++,从 Java,OC 到 Swift,Kotlin,甚至被纳入教材的 Python,都有一个共同的特点。随着 CPU 性能的越来越强悍,提高生产力似乎都成了现代高级编程语言的共同目标。Kotlin 就是一个好例子,做同样的事情,完成同样的功能,Java 的确需要更多的代码,Kotlin 也的确给 Android 开发提升了效率。特别是在异步任务方面,Kotlin 提供了协程,而这是 Java 所不具备的。
老版本 Retrofit 的使用
在介绍如何使用 Retrofit 2.6.0
之前,我们先来看一下老版本的 Retrofit 是如何基于 Kotlin Coroutines 工作的,以登录接口为例。
首先在 WanService 接口中作如下定义:
@POST("/user/login")
fun login(@Field("username") userName: String,
@Field("password") passWord: String): Deferred<WanResponse<User>>
注意这里使用的返回值是 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<User> {
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<User>
看到区别了吗?首先,不再返回 Deferred
对象,而是直接返回我们需要的 WanResponse
对象。其次,使用了 suspend
来修改方法,标记这是挂起函数。
第三步,修改 LoginRepository
中方法定义。
suspend fun login(userName: String, passWord: String): WanResponse<User> {
return apiCall { WanRetrofitClient.service.login(userName, passWord) }
}
与之前的版本相比,这里不需要调用 await
方法了。其实并不是不调用了,而是 Retrofit
帮助我们自动调用了。
最后别忘了去除之前添加的 kotlin-coroutines-adapter
,因为我们不再需要人工返回 Deferred
对象,也不再需要手动调用 await
了。
...
//.addCallAdapterFactory(CoroutineCallAdapterFactory.invoke())
...
至此,基于 Retrofit 2.6.0
版本的改造就已经完成了。