两种给给LiveData添加请求状态回调的方法

最近开发中一直在使用ViewModel +LiveData 配合来实现MVVM,LiveData 这个东西确实挺好用的.但是存在一个问题,看代码:

//表示一个从后台拿到的User类型来显示到UI
val  user  = MultableLiveData<User>()

user 从后台来,就可能出错,单单用一个user 是无法通知到UI层出错的,这时候需要一个变量来标记出错的内容,同时如果UI层想要在开始请求时显示加载动画,就又需要一个 变量来标记开始,这时候代码就变成了这样

val  user  = MultableLiveData<User>()
var startState = MultableLiveData<Any>()
var errorState = MultableLiveData<Throwable>()

然后在请求网络的回调中,开始请求数据就这样(为了容易理解,网络请求返回采用rxjava 的subscribe 的回调格式,所以就不要好奇为什么不直接在view 层订阅 网络请求返回的Observable 了,因为没有使用rxjava):

requetUser().subscribe{
      onStart = {startState.value = Any()},
      onNext = {user.postValue(it)},
      onError = {user.postValue(it)}
}

然后在 view层分别观察这三个状态:

user.observe(this,Observer{ 回显...})
startState.observe(this,Observer{开始加载动画....})
errorState.observe(this,Observer{提示出错....})

这样清晰是挺清晰的.如果一个页面对应一个接口还好.但是如果是很复杂的页面对应多个接口,而且每个接口的开始动画和出错提示是分开的,那么我们就需要给每一个接口创建三个liveData,这样就会有非常多的livedata字段..为了少写几个字段,我就想能不能制造一个天生的有开始状态,出错状态,正常状态且具有livedata特性的东东呢,这样一个接口对应一个livedata 字段就好了,于是我就做了如下的封装

import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.LiveData
import androidx.lifecycle.MutableLiveData
import androidx.lifecycle.Observer

class StateLiveData<T>(value: T?) {
    private val liveData = MutableLiveData<T>(value)
    private var onStart = MutableLiveData<Any>()
    private var onError = MutableLiveData<Throwable>()

    var value: T?
        get() {
            return liveData.value
        }
        set(value) {
            liveData.value = value
        }

    fun postValue(value: T?) {
        liveData.postValue(value)
    }


    fun onStart() {
        onStart.value = Any()
    }

    fun postOnStart() {
        onStart.postValue(Any())
    }

    fun onError(throwable: Throwable) {
        onError.postValue(throwable)
    }


    fun observe(
        lifecycleOwner: LifecycleOwner,
        onNext: OnNext<T>? = null,
        onStart: OnStart? = null,
        onError: OnError? = null
    ) {
        liveData.observe(lifecycleOwner, Observer {
            onNext?.invoke(it)

        })
        this.onError.observe(lifecycleOwner, Observer {
            onError?.invoke(it)
        })

        this.onStart.observe(lifecycleOwner, Observer {
            onStart?.invoke()
        })

    }

}

typealias  OnError = (Throwable) -> Unit

typealias OnStart = () -> Unit

typealias OnNext<T> = (T) -> Unit

每一个 StateLiveData都具有 开始状态,出错状态,正常状态 这时候我们就可以这样写了

val user = StateLiveData<User>()

然后这样处理网络请求回调:

requestUser().subscribe(
onStart = { user.onStart() },
onNext = {user.value = it},
onError = {user.onError(it) }
)

然后在view层这样观测user

user.observe(this,
onNext = {回显结果...},
onStart = {加载动画..},
onError = {错误提示...}
)

这样就用构造了一个具备状态的livedata了..能够少写很多字段了哈哈哈...
这个是简单的封装,如果有需要ObserverForever 方法,对着这些方法进行一一封装就好了

另外想少写字段的方法还有一种就是构建一个带数据状态的UiState

sealed class UiState() {
    class Start : UiState()
    class Error(val throwable: Throwable) : UiState()
    class Next<T>(val t: T) : UiState()
}

然后User字段这样声明

val user = MultableLiveData<UiState<User>>

然后这样观测请求结果

requestUser().subscribe(
onStart = {user.postValue(UiState.Start()) },
onNext = { user.postValue(UiState.Next(it))},
onError = {user.postValue(UiState.Error(it) }
)

在view层这样观察

 user.observe(this, Observer{ 
            when(it){
                is UiState.Start -> {加载动画...}
                is UiState.Error ->{错误提示...}
                is UiState.Next<*> -> {
                    it.t as User {回显结果...}
                }
            }
        })
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容