写点什么

Kotlin-+- 协程 -+-Retrofit-,记录一次腾讯 Android 岗面试笔试总结

用户头像
Android架构
关注
发布于: 7 小时前
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.simpleNameprivate 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


《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
浏览器打开:qq.cn.hn/FTe 免费领取
复制代码


.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


private val datas: MutableLiveData<List<Data>> by lazy { MutableLiveData<List<Data>>().also { loadDatas() } }


private val repository = ArticleRepository()fun getActicle(): LiveData<List<Data>> {return datas}


private fun loadDatas() {launchUI {Log.i(TAG,"loadDatas1 run in {Thread.currentThread().name}")datas.value = result.data}// Do an asynchronous operation to fetch users.}}


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


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


看到了吧,各司其职,效果很棒

异常处理

搞了半天才发现没有弄异常处理,当请求失败之后,项目就崩溃了,这不是是我们想要的结果,由于好没有想到更好的处理方式,只能在外面套个 tyr catch 顶一顶了,参考如下


open class BaseViewModel : ViewModel(), LifecycleObserver{private val viewModelJob = SupervisorJob()private val uiScope = CoroutineScope(Dispatchers.Main + viewModelJob)private val error by lazy { MutableLiveData<Exception>() }private val finally by lazy { MutableLiveData<Int>() }//运行在 UI 线程的协程 fun launchUI( block: suspend CoroutineScope.() -> Unit) {uiScope.launch(Dispatchers.Main) {try {block()}catch (e:Exception){error.value = e}finally {finally.value = 200}}

用户头像

Android架构

关注

还未添加个人签名 2021.10.31 加入

还未添加个人简介

评论

发布
暂无评论
Kotlin-+-协程-+-Retrofit-,记录一次腾讯Android岗面试笔试总结