Retrofit 2.6.0 ! 更快捷的协程体验 !
Posted u012551350
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Retrofit 2.6.0 ! 更快捷的协程体验 !相关的知识,希望对你有一定的参考价值。
本文作者: 秉心说
本文链接:
https://juejin.im/user/586eff908d6d81005879507d
文末有彩蛋
近日 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
回到正题,本篇主要介绍 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<WanResponse<User>>
fun login(@Field("username") userName: String,
@Field("password") passWord: String): Deferred<WanResponse<User>>
注意这里使用的返回值是 Deferred<T>
对象,这就意味着使用的时候要通过 await
来获取返回值。那么如何让 Retrofit 直接返回 Deferred<T>
呢?使用的也是 JakeWharton 的开源库:
implementation 'com.jakewharton.retrofit:retrofit2-kotlin-coroutines-adapter:0.9.2''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()
return apiCall WanRetrofitClient.service.login(userName, passWord).await()
这里使用 await
来获取 Deferred<T>
的返回值。
最后在 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
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''com.squareup.retrofit2:retrofit:2.6.0'
第二步,修改 WanService
中接口的定义。
@POST("/user/login")suspend fun login(@Field("username") userName: String, @Field("password") passWord: String): WanResponse<User>
suspend fun login(@Field("username") userName: String,
@Field("password") passWord: String): WanResponse<User>
看到区别了吗?首先,不再返回 Deferred<T>
对象,而是直接返回我们需要的 WanResponse
对象。其次,使用了 suspend
来修改方法,标记这是挂起函数。
第三步,修改 LoginRepository
中方法定义。
suspend fun login(userName: String, passWord: String): WanResponse<User> return apiCall WanRetrofitClient.service.login(userName, passWord)
return apiCall WanRetrofitClient.service.login(userName, passWord)
与之前的版本相比,这里不需要调用 await
方法了。其实并不是不调用了,而是 Retrofit 帮助我们自动调用了。
最后别忘了去除之前添加的 kotlin-coroutines-adapter
,因为我们不再需要人工返回 Deferred<T>
对象,也不再需要手动调用 await
了。
...//.addCallAdapterFactory(CoroutineCallAdapterFactory.invoke())...//.addCallAdapterFactory(CoroutineCallAdapterFactory.invoke())
...
至此,基于 Retrofit 2.6.0
版本的改造就已经完成了。我的 Wanandroid 项目已经完成全部修改,具体修改内容可见 commit。
总结
随着 Kotlin 成为 Android 开发的首选语言,越来越多的新特性都将在 Kotlin 上优先实现。协程作为 Kotlin 的异步利器,很值得我们学习。如果你还没有入手,那么,从我的 Wanandroid 开始吧 !
请在后台回复「抽奖」参与抽奖
请在后台回复「抽奖」参与抽奖
请在后台回复「抽奖」参与抽奖
以上是关于Retrofit 2.6.0 ! 更快捷的协程体验 !的主要内容,如果未能解决你的问题,请参考以下文章