Kotlin + 协程 + Retrofit + MVVM优雅的实现网络请求

override fun onResponse(call: retrofit2.Call, response: Response) {

TODO(“not implemented”) //To change body of created functions use File | Settings | File Templates.

}

})








这只是描述了一个retrofit的简单请求方式,实际项目中基本上都会封装之后再使用,也为了提高代码的可读性,降低各部分的耦合性, 通俗点来说,只有各司其职才能把工作干好嘛,接下来咱们就围绕着各司其职来一个一个实现



协程实现

----



接下来把上面的请求换成协程的方式来实现



### 1.创建RetrofitClient





`object为了使RetrofitClient 只能有一个实例 ~~~     object RetrofitClient {         val BASE_URL =  "https://wanandroid.com/"         val reqApi by lazy {             val retrofit = Retrofit.Builder()                     .baseUrl(BASE_URL)                     .addConverterFactory(GsonConverterFactory.create())                     .addCallAdapterFactory(CoroutineCallAdapterFactory())                     .build()             return@lazy retrofit.create(RequestService::class.java)         }     } ~~~`







### 2.创建service接口类





`~~~ interface RequestService {     @GET("wxarticle/chapters/json")     fun getDatas() : Deferred<DataBean> } ~~~`







因为我们后续会使用到协程,所以这儿将Call换成了Deferred



### 3.发起请求





`~~~     GlobalScope.launch(Dispatchers.Main) {         withContext(Dispatchers.IO){           val dataBean = RetrofitClient.reqApi.getDatas().await()         }         //更新ui     } ~~~`







上面用到了协程,这儿只讲述他的应用了,具体的移步官方文档进一步了解。 网络请求在协程中,并且在IO调度单元,所以不用担会阻塞主线程



协程 + ViewModel + LiveData实现

---------------------------



上面也只是简单的实现,只不过是换成了协程,在项目中,还可以进一步封装,方便使用前面也提到了MVVM,所以还用到了Android 新引入的组件架构之ViewModel和LiveData,先看ViewModel的实现





`class ScrollingViewModel  : ViewModel() {     private val TAG = ScrollingViewModel::class.java.simpleName     private val datas: MutableLiveData<DataBean> by lazy { MutableLiveData<DataBean>().also { loadDatas() } }     private val repository = ArticleRepository()     fun getActicle(): LiveData<DataBean> {         return datas     }     private fun loadDatas() {         GlobalScope.launch(Dispatchers.Main) {             getData()         }         // Do an asynchronous operation to fetch users.     }     private suspend fun getData() {         val result = withContext(Dispatchers.IO){ //            delay(10000)             repository.getDatas()         }        datas.value = result     } }`







ViewModel将作为View与数据的中间人,Repository专职数据获取,下面看一下Repository的代码,用来发起网络请求获取数据





`class ArticleRepository {      suspend fun getDatas(): DataBean {           return RetrofitClient.reqApi.getDatas().await()       }   }`







在Activity中代码如下





`private fun initData() {         model.getActicle().observe(this, Observer{             //获取到数据             toolbar.setBackgroundColor(Color.RED)         })     }`







后续优化

----



### 1.内存泄漏问题解决方案



结和了各位大佬们的意见,将使用GlobalScope可能会出现内存泄漏的问题进行了优化。因为在协程进行请求的过程中,若此时ViewModel销毁,里面的协程正在请求的话,将无法销毁,出现内存泄漏,所以在ViewModel onCleared 里面,即使结束协程任务,参考代码如下。





`open class BaseViewModel : ViewModel(), LifecycleObserver{     private val viewModelJob = SupervisorJob()     private val uiScope = CoroutineScope(Dispatchers.Main + viewModelJob)     //运行在UI线程的协程     fun launchUI( block: suspend CoroutineScope.() -> Unit) {         try {             uiScope.launch(Dispatchers.Main) {                 block()             }         }catch (e:Exception){             e.printStackTrace()         }     }     override fun onCleared() {         super.onCleared()         viewModelJob.cancel()     } }`







当然,最好的方式是使用viewModelScope,但是我在引入该包的时候,会报错,由于最近比较忙暂时还没来得急解决,后续问题有时间我也会继续修改,还望各位大佬能帮忙指点



### 2.优化请求代码



先看下之前的请求代码





`private suspend fun getData() {         val result = withContext(Dispatchers.IO){ //            delay(10000)             repository.getDatas()         }        datas.value = result     }`







每一次都需要写个withContext(),实际运用中,感觉有点不方便,于是乎想了一下,怎么才能给他封进请求方法里面? 代码如下





open class BaseRepository {

suspend fun <T : Any> request(call: suspend () -> ResponseData<T>): ResponseData<T> {

return withContext(Dispatchers.IO){ call.invoke()}

}

}







通过在BaseRepository里面写了一个专门的请求方法,这样每次只需执行request就行了 请求参考如下





`class ArticleRepository : BaseRepository() {     suspend fun getDatas(): ResponseData<List<Data>> {        return request {            delay(10000)            Log.i(ScrollingViewModel::class.java.simpleName,"loadDatas1 run in  ${Thread.currentThread().name}")            RetrofitClient.reqApi.getDatas().await() }     } }`







注:这个 delay(10000)只是我测试用的,意思是休眠当前协程,防止萌新在自己项目中加上了,还是有必要说一下的



再看看ViewModel中就太简单了





`` `class ScrollingViewModel : BaseViewModel() {     private val TAG = ScrollingViewModel::class.java.simpleName` ``



`<span class="hljs-keyword">private</span> val datas: MutableLiveData&lt;List&lt;Data&gt;&gt; by lazy { MutableLiveData&lt;List&lt;Data&gt;&gt;().also { loadDatas() } }  <span class="hljs-keyword">private</span> val repository = ArticleRepository() fun getActicle(): LiveData&lt;List&lt;Data&gt;&gt; {     <span class="hljs-keyword">return</span> datas }  <span class="hljs-keyword">private</span> fun loadDatas() {     launchUI {         Log.i(TAG,<span class="hljs-string">"loadDatas1 run in  ${Thread.currentThread().name}"</span>)         val result = repository.getDatas()         Log.i(TAG,<span class="hljs-string">"loadDatas3 run in  ${Thread.currentThread().name}"</span>)         datas.value = result.data     }     <span class="hljs-comment">// Do an asynchronous operation to fetch users.</span> }`











}  



注意看请求部分,就两句话,一句发起请求val result = repository.getDatas(),然后就是为我们的LiveData赋值了,看起有没有同步代码的感觉,这就是协程的魅力所在,为了验证我们的请求没有阻塞主线程,我打印了日志





`06-19 12:26:35.736 13648-13648/huaan.com.mvvmdemo I/ScrollingViewModel: loadDatas start run in  main 06-19 12:26:45.743 13648-13684/huaan.com.mvvmdemo I/ScrollingViewModel: request run in  DefaultDispatcher-worker-1 06-19 12:26:46.227 13648-13648/huaan.com.mvvmdemo I/ScrollingViewModel: loadDatas end  run in  main`








**自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。**

**深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!**

**因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。**
![img](https://img-blog.csdnimg.cn/img_convert/22d5dd893f322dde5f7ca89702602d20.png)
![img](https://img-blog.csdnimg.cn/img_convert/dbbf4d37bdedbf32aad5425c4de9fbea.png)
![img](https://img-blog.csdnimg.cn/img_convert/6d04ca122f9b83a54a5a721dccd82b2d.png)
![img](https://img-blog.csdnimg.cn/img_convert/46fe157406202dfdfab9da12db947b78.png)
![img](https://img-blog.csdnimg.cn/img_convert/b2304d57ccd5b1e7470a135b4e7a1695.png)
![img](https://img-blog.csdnimg.cn/img_convert/36134c922c6abeaed3a37cd773f0e5cd.png)
![img](https://img-blog.csdnimg.cn/13f2cb2e05a14868a3f0fd6ac81d625c.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!**

**由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新**

**如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)**
![img](https://img-blog.csdnimg.cn/img_convert/b756c27add92f66674ca50f840d32e04.png)



### 关于面试的充分准备

一些基础知识和理论肯定是要背的,要理解的背,用自己的语言总结一下背下来。

虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,**现在高级工程师还是比较缺少的**,我能明显感觉到国庆后多了很多高级职位,所以努力让自己成为高级工程师才是最重要的。

好了,希望对大家有所帮助。

接下来是整理的一些Android学习资料,**有兴趣的朋友们可以关注下我免费领取方式**。

**①Android开发核心知识点笔记**

**②对标“阿里 P7” 40W+年薪企业资深架构师成长学习路线图**

![](https://img-blog.csdnimg.cn/img_convert/0074f73fd6219cc645b4cb0e77c45831.webp?x-oss-process=image/format,png)

**③面试精品集锦汇总**

![](https://img-blog.csdnimg.cn/img_convert/81d3eaa04f64aa1bcf3cf7bf8d187370.webp?x-oss-process=image/format,png)

**④全套体系化高级架构视频**

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!

![](https://img-blog.csdnimg.cn/img_convert/0a4271082095c63853e5ea82104cd909.webp?x-oss-process=image/format,png)

-pG028Dvk-1711963686431)]

**④全套体系化高级架构视频**

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!

[外链图片转存中...(img-mopMcr94-1711963686432)]

> **本文已被[CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》]( )收录**
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值