网站首页 > 技术教程 正文
前言
最近很不顺利,每天晚上回家都打不到车!之前晚上10点很容易叫到车,不是说好996福报的么?难不成大家都在享受福报,司机都忙不过来了?管他呢,就算打不到车,我也要学习,毕竟一天不学习我浑身难受!
之前几篇文章聊过JatPack中LiveData和ViewModel的基本使用和原理。历史文章如下:
今天咱们继续看一下实际的应用。实战篇初步打算俩篇文章,分别是:
- Google Sample写的一个简易的网络框架:NetworkBoundResource。
- MVVM项目实战
NetworkBoundResource篇
一、什么是NetworkBoundResource
首先来说一下 NetworkBoundResource是什么,其实 NetworkBoundResource单纯就是一个类,全类也就100+行,但是这个类结合LiveData,创造了极为便利的常用网络功能,比如:
- 不请求网络,直接使用缓存
- 自定义策略,是否请求网络
- 网络加载失败后使用缓存
- 返回类型处理
- 等等
不说了,直接上代码!先看一段这个类的使用:
// UI层直接调用这个方法,拿到LiveData,监听即可(当然,正常来说需要设计一番,UI层直接粗暴的调用,不大合适~) fun loadData(queryId: Long = -1): LiveData<Resource<DataResp>> { return object : NetworkBoundResource<DataResp, DataResp>( appExecutors ) { override fun saveCallResult(item: DataResp) { // 此方法,在网络数据回来后调用,我们可以做一些持久化的逻辑 } override fun shouldFetch(data: DataResp?): Boolean { // 自己控制,是否触发网络请求,如果false,则调用loadFromDb() return isUseNetWork } override fun loadFromDb(): LiveData<MusicStoreMainResp> { // 自己实现从非网络环境下获取数据的逻辑(比如内存,DB) return data } override fun createCall(): LiveData<ApiResponse<DataResp>> { return }.asLiveData() }
我们可以看到,4个实现方法,分别对应了:
- 数据返回后的持久化回调
- 是否走网络请求
- 从本地请求数据(业务方自己实现)
- 网络请求(业务方自己实现)
对于我们业务方来说,只需要调用oadData(),然后observe(),返回的LiveData即可。
当然对应的正真的业务请求需要自己实现
二、NetworkBoundResource流程图
也就是说NetworkBoundResource帮我们抽象了一系列的逻辑,而且,它的实现非常的短,让我们来看一下代码,NetworkBoundResource做了什么?能帮我们如此简单的完成这么多逻辑?
三、NetworkBoundResource源码实现
上源码:
详细使用可以参考Google Sample:https://github.com/googlesamples/android-architecture-components/tree/master/GithubBrowserSample
abstract class NetworkBoundResource @MainThread constructor(private val appExecutors: AppExecutors) { //这里是业务能拿到的数据,livedata //MediatorLiveData不多说了吧,上文已经介绍过了 private val result = MediatorLiveData<Resource<ResultType>>() init { // 先发一个LOADIN,通知业务放处理LOADING态 result.value = Resource.loading(null) @Suppress("LeakingThis") //db也是一个数据源 val dbSource = loadFromDb() result.addSource(dbSource) { data -> //db的第一次回调,是用来判断数据有效期的 result.removeSource(dbSource) //是否有效,业务自行定义(请求网络的策略) if (shouldFetch(data)) { fetchFromNetwork(dbSource) } else { //数据有效,重新观察一次,观察者会立马收到一次回调{Source.plug} result.addSource(dbSource) { newData -> setValue(Resource.success(newData)) } } } } @MainThread private fun setValue(newValue: Resource<ResultType>) { if (result.value != newValue) { result.value = newValue } } private fun fetchFromNetwork(dbSource: LiveData<ResultType>) { val apiResponse = createCall() // 将dbsource重新add,它将快速地发送其最新值。db有数据,但是过期了,先回调给业务展示 // 这里保证了,LOADING态时也可以拿到数据,并展示给用户。 result.addSource(dbSource) { newData -> setValue(Resource.loading(newData)) } result.addSource(apiResponse) { response -> //这里又是用来控制流程,移除,避免数据乱入,而且设计者不让add重复的source result.removeSource(apiResponse) result.removeSource(dbSource) when (response) { is ApiSuccessResponse -> { appExecutors.diskIO.execute { //数据回来先存缓存,这样我们下次请求过来时,可能保证LOADING态拿到的数据是最新的。 saveCallResult(processResponse(response)) appExecutors.mainThread.execute { // 原注释:we specially request a new live data, // otherwise we will get immediately last cached value, // which may not be updated with latest results received from network. //重新从库里面读取 result.addSource(loadFromDb()) { newData -> setValue(Resource.success(newData)) } } } } is ApiEmptyResponse -> { appExecutors.mainThread.execute { // reload from disk whatever we had result.addSource(loadFromDb()) { newData -> setValue(Resource.success(newData)) } } } is ApiErrorResponse -> { onFetchFailed() result.addSource(dbSource) { newData -> setValue(Resource.error(response.exception, newData)) } } } } } // 业务方自行处理的抽象方法 protected open fun onFetchFailed() {} fun asLiveData() = result as LiveData<Resource<ResultType>> @WorkerThread protected open fun processResponse(response: ApiSuccessResponse<RequestType>) = response.data @WorkerThread protected abstract fun saveCallResult(item: RequestType) @MainThread protected abstract fun shouldFetch(data: ResultType?): Boolean @MainThread protected abstract fun loadFromDb(): LiveData<ResultType> @MainThread protected abstract fun createCall(): LiveData<ApiResponse<RequestType>> }
有朋友可能会问ApiResponse是啥,很简单就是这个:
尾声
这一部分,建议大家好好理解一下。因为真的真的真的很好用,它的设计结合了LiveData一系列的巧妙应用。理解之后,大家绝对会对LiveData有更加深入的理解,并且在接下来的MVVM中,也会感受到这其中的巧妙和爽快。
接下来的实战篇,基本就是结合NetworkBoundResource的MVVM设计,希望能够给大家在业务架构上带来帮助。
猜你喜欢
- 2024-10-03 《朗读者》——董卿(主编) 朗读者董卿好看视频
- 2024-10-03 60部永远消失在电视上的动画片 60部永远消失在电视上的动画电影
- 2024-10-03 100多种常见中成药,对应240多种常见疾病,此文到手,买药不愁
- 2024-10-03 Clubhouse一周内在Android系统上突破了100万次下载大关
- 2024-10-03 「新品资讯」Astell&Kern发布TWS真无线耳机AK UW100
- 2024-10-03 「图」Android端Edge发现新彩蛋:标签页超100计数显示Ninja Cat Emoji
- 2024-10-03 Android安全计划帮助修复了Google Play中的超过100万个问题应用
- 2024-10-03 八大人际定律,教你如何快速获得“好人缘”
- 2024-10-03 那一年,我高考落榜了 那一年我们高考
- 2024-10-03 Shopify适用于Android 的 TSP100IIILAN 和 TSP100IIIW 收据打印机
你 发表评论:
欢迎- 01-11关于Vmware workstation的网络设置
- 01-11使用VMware Workstation虚拟机安装Windows 10详细教程
- 01-11VMware Workstation 17.5.1 Pro for Windows & Linux - 桌面虚拟化软件
- 01-11VMware? Workstation 17 Pro软件中,如何显示 “我的计算机” 选项卡
- 01-11银河麒麟桌面操作系统安装VMware workstation pro
- 01-11VmwareWorkstation17.6安装windows7x64虚拟机后安装vmtools
- 01-11发布VMware Workstation Pro 17.0稳定版
- 01-11VMware Workstation安装ESXi 7安装篇
- 最近发表
-
- 关于Vmware workstation的网络设置
- 使用VMware Workstation虚拟机安装Windows 10详细教程
- VMware Workstation 17.5.1 Pro for Windows & Linux - 桌面虚拟化软件
- VMware? Workstation 17 Pro软件中,如何显示 “我的计算机” 选项卡
- 银河麒麟桌面操作系统安装VMware workstation pro
- VmwareWorkstation17.6安装windows7x64虚拟机后安装vmtools
- 发布VMware Workstation Pro 17.0稳定版
- VMware Workstation安装ESXi 7安装篇
- 如何安装虚拟机(vmware workstation)
- VMware Workstation之网络配置
- 标签列表
-
- sd分区 (65)
- raid5数据恢复 (81)
- 地址转换 (73)
- 手机存储卡根目录 (55)
- tcp端口 (74)
- project server (59)
- 双击ctrl (55)
- 鼠标 单击变双击 (67)
- debugview (59)
- 字符动画 (65)
- flushdns (57)
- ps复制快捷键 (57)
- 清除系统垃圾代码 (58)
- web服务器的架设 (67)
- 16进制转换 (69)
- xclient (55)
- ps源文件 (67)
- filezilla server (59)
- 句柄无效 (56)
- word页眉页脚设置 (59)
- ansys实例 (56)
- 6 1 3固件 (59)
- sqlserver2000挂起 (59)
- vm虚拟主机 (55)
- config (61)
本文暂时没有评论,来添加一个吧(●'◡'●)