百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

作为android开发者,你知Google提供的这个100行的小框架吗?

ccwgpt 2024-09-20 13:07 33 浏览 0 评论

前言

最近很不顺利,每天晚上回家都打不到车!之前晚上10点很容易叫到车,不是说好996福报的么?难不成大家都在享受福报,司机都忙不过来了?管他呢,就算打不到车,我也要学习,毕竟一天不学习我浑身难受!

之前几篇文章聊过JatPack中LiveData和ViewModel的基本使用和原理。历史文章如下:

一点点入坑JetPack:Lifecycle篇

一点点入坑JetPack:Lifecycle源码篇

Google大费周折推的ViewModel是什么?

只需三分钟,你也会和我一样爱上:LiveData

今天咱们继续看一下实际的应用。实战篇初步打算俩篇文章,分别是:

  • 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设计,希望能够给大家在业务架构上带来帮助。

相关推荐

用Deepseek扩写土木工程毕业论文实操指南

用Deepseek扩写毕业论文实操指南一、前期准备整理现有论文初稿/提纲列清楚论文核心框架(背景、现状、意义、方法、数据、结论等)梳理好关键文献,明确核心技术路线二、Deepseek扩写核心思路...

985学霸亲授,DeepSeek也能绘6大科研图表,5分钟就出图

在实验数据处理中,高效可视化是每个科研人的必修课。传统绘图软件操作复杂、耗时费力,而智能工具DeepSeek的出现彻底改变了这一现状。本文将详解如何用DeepSeek一键生成六大科研常用图表,从思维导...

AI写论文刷屏?大学生正在丢掉的思考力

一、宿舍深夜:当论文变成"Ctrl+C+V"凌晨两点的大学宿舍,小王对着电脑屏幕叹气。本该三天前开始写的近代史论文,此刻还一片空白。他熟练打开某AI写作网站,输入"论五四运动的...

Grok在辅助论文写作上能不能既“聪明”又“可怕”?!

AcademicIdeas-学境思源AI初稿写作随着人工智能技术的飞速发展,论文写作这一学术任务正迎来新的助力。2025年2月18日,美国xAI公司推出了备受瞩目的Grok3模型,其创始人埃隆·...

大四论文沟通场景!音频转文字难题听脑AI来化解

大四学生都知道,写论文时和导师沟通修改意见,简直是“过关斩将”。电话、语音沟通完,想把导师说的修改方向、重点要求记下来,麻烦事儿可不少。手写记不全,用普通录音转文字工具,转完还得自己慢慢找重点,稍不注...

论文写作 | 技术路线图怎么画?(提供经典优秀模板参考)

技术路线图是一种图表或文字说明,用于描述研究目标、方法和实施计划。它展示了研究的整体框架和步骤,有助于读者理解研究的逻辑和进展。在课题及论文中,技术路线图是常见的一部分,甚至是一个类似心脏一样的中枢器...

25年信息系统项目管理师考试第2批论文题目写作建议思路框架

25年信息系统项目管理师考试第2批论文题目写作建议思路框架--马军老师

微信购物应尽快纳入法律框架(微信购物管辖)

符向军近日,甘肃省工商行政管理局发布《2016年上半年信息分析报告》。报告显示,微信网购纠纷迅猛增长,网络购物投诉呈上升趋势。投诉的主要问题有出售的商品质量不过关、消费者通过微信付款后对方不发货、购买...

泛珠三角区域网络媒体与腾讯微信签署《战略合作框架协议》

新海南客户端、南海网7月14日消息(记者任桐)7月14日上午,参加第四届泛珠三角区域合作网络媒体论坛的区域网络媒体负责人及嘉宾一行到腾讯微信总部座谈交流,并签署《战略合作框架协议》(以下简称《框架协...

离线使用、植入微信-看乐心Mambo手环如何打破框架

从2014年开始智能手环就成功进入人们的生活,至今已经演变出数据监测、信息推送、心率监测等诸多五花八门的功能,人们选择智能手环并不指望其能够改变身体健康情况,更多的是通过数据来正视自身运动情况和身体健...

微信私域电商运营策略与框架(微信私域怎么做)

...

华专网络:如何零基础制作一个网站出来?

#如何零基础制作一个网站出来?#你是不是觉得网站建设很复杂,觉得自己是小白,需求不明确、流程搞不懂、怕被外包公司坑……这些问题我都懂!今天华专网络就用大白话给你捋清楚建站的全流程,让你轻松get网站制...

WAIC2024丨明日上午9点,不见不散!共同探讨智能社会与全球治理框架

大咖云集,硕果闪耀WAIC2024世界人工智能大会智能社会论坛将于7月5日9:00-12:00与你相约直播间WAIC2024上海杨浦同济大学哔哩哔哩多平台同步直播探讨智能社会与全球治理框架WAIC...

约基奇:森林狼换来戈贝尔时大家都在嘲笑 他们的阵容框架很不错

直播吧5月4日讯西部季后赛半决赛,掘金将迎战森林狼,约基奇赛前接受采访。约基奇说道:“当蒂姆-康纳利(森林狼总经理、前掘金总经理&曾选中约基奇)做了那笔交易(换来戈贝尔)时,每个人都在嘲笑他...

视频号带货为什么一个流量都没有?顶级分析框架送给你

视频号带货为什么一个流量都没有?遇到问题,一定是步步来分析内容,视频号带货一个流量都没有,用另外一个意思来讲,就可以说是零播放。为什么视频号带货一个流量都没有?跟你说再多,都不如来个分析框架。1、是否...

取消回复欢迎 发表评论: