分享android mvvm 实践

网上看了好多的android mvvm形式,大多数都很复杂,不够简洁,造成项目代码臃肿,逻辑难以梳理,因此分享下自己的mvvm实践(大多数来源于优秀项目思路的整合,可能不是最优的):
1.第一步,当然是接口了(kt代码)

interface ApiService {
    companion object {
        val instance by lazy { RetrofitFactory.create(ApiService::class.java) }
    }

    @Headers("$DOMAIN_NAME_HEADER$BASE_HTTP_URL_NAME")
    @POST("user/login")
    @FormUrlEncoded
    suspend fun login(@FieldMap map: Map<String, String>): ResponseBean<UserBean>
}

其中RetrofitFactory代码为:

object RetrofitFactory {
    //初始化
    //通用拦截
    private val interceptor: Interceptor by lazy {
        Interceptor { chain ->
            val request = chain.request()
            val builder = request.newBuilder()
            builder.addHeader("X-Client-Platform", "Android")
                .addHeader("X-Client-Version", BuildConfig.VERSION_NAME)
                .addHeader("X-Client-Build", BuildConfig.VERSION_CODE.toString())
                .build()
            chain.proceed(request)
        }
    }

    //Retrofit实例化
    private val retrofit: Retrofit by lazy {
        Retrofit.Builder()
            .baseUrl(Constant.DEFAULT_URL)
            .addConverterFactory(NullOnEmptyConverterFactory())
            .addConverterFactory(CustomConverterFactory.create(ResponseBean::class.java))
            .client(RetrofitUrlManager.getInstance().with(initClient()).build())
            .build()
    }

    /*
        OKHttp创建
     */
    private fun initClient(): OkHttpClient.Builder {
        val sslParams1 = HttpsUtils.getSslSocketFactory()
        return OkHttpClient.Builder()
            .sslSocketFactory(sslParams1.sSLSocketFactory, sslParams1.trustManager)
            .addInterceptor(initLogInterceptor())
            .addInterceptor(interceptor)
            .connectTimeout(30, TimeUnit.SECONDS)
            .readTimeout(30, TimeUnit.SECONDS)

    }

    /*
        日志拦截器
     */
    private fun initLogInterceptor(): HttpLoggingInterceptor {
        val interceptor = HttpLoggingInterceptor()
        interceptor.setPrintLevel(HttpLoggingInterceptor.Level.BODY)
        interceptor.setColorLevel(Level.INFO)
        return interceptor
    }

    /*
        具体服务实例化
     */
    fun <T> create(service: Class<T>): T {
        return retrofit.create(service)
    }

}

其中header用到了me.jessyan:retrofit-url-manager:1.4.0这个库,主要作用是多域名配置
2.第二步

object Repository : BaseRepository() {
    suspend fun login(map: Map<String, String>): ResponseBean<UserBean> {
        return apiCall { ApiService.instance.login(map) }
    }
}

其中用到的BaseRepository代码为:

open class BaseRepository {
    suspend fun <T> apiCall(call: suspend () -> ResponseBean<T>): ResponseBean<T> {
        return call.invoke()
    }


    suspend fun <T> dbCall(call: suspend () -> T): T {
        return call.invoke()
    }

}

3.第三步,就是viewmodel了,代码如下

class LoginViewModel: BaseViewModel() {
    val mLoginLiveData = StateLiveData<UserBean>()
    fun login(userName:String,password:String){
        if (userName.isEmpty()){
            mLoginLiveData.postError(1,"用户名不能为空")
            return
        }
        if (password.isEmpty()){
            mLoginLiveData.postError(1,"密码不能为空")
            return
        }
        val map = hashMapOf<String, String>()
        map["adminLoginName"] = userName
        map["adminPassword"] = password
        launch(mLoginLiveData){
            await { Repository.login(map) }
        }

    }
}

其中用到的BaseViewModel,代码为:

open class BaseViewModel : ViewModel() {
    fun launch(block: suspend CoroutineScope.() -> Unit) {
        if (!isNetUsable)
            return
        viewModelScope.launch { block() }
    }

    fun <T> launch(
        liveData: StateLiveData<T>,
        tryBlock: suspend CoroutineScope.() -> T
    ) {
        if (!isNetUsable) {
            liveData.postStart()
            liveData.postComplete()
            liveData.postNetError()
            return
        }
        launch {
            tryCatch(liveData, tryBlock)
        }
    }

    private suspend fun <T> tryCatch(
        liveData: StateLiveData<T>,
        tryBlock: suspend CoroutineScope.() -> T
    ) {
        coroutineScope {
            try {
                d("start")
                liveData.postStart()
                d("start-end")
                val response = tryBlock.invoke(this)
                d("success")
                liveData.value = response
                d("parse")
            } catch (e: OperatorException) {
                e.printStackTrace()
                d("fail")
                liveData.postError(e.code, e.msg)
            } catch (e: Exception) {
                if (isDebug) {
                    throw e
                } else {
                    liveData.postError(1, "网络连接失败,请稍候重试!")
                    e.message?.let { d(it) }
                    CrashReport.postCatchedException(e)
                }
            } finally {
                liveData.postComplete()
                d("complete")
            }
        }
    }

}

StateLiveData代码如下:

class StateLiveData<T> : MutableLiveData<T>() {
    val startState = MutableLiveData<Int>()
    val otherState = MutableLiveData<OtherState>()
    val completeState = MutableLiveData<Int>()
    val error = MutableLiveData<Pair<Int, String?>>()

    fun postStart() {
        if (startState.value != null)
            startState.value = startState.value!! + 1
        else
            startState.value = 1
    }

    fun postComplete() {
        if (completeState.value != null)
            completeState.value = completeState.value!! + 1
        else
            completeState.value = 1
    }

    fun postEmpty() {
        otherState.value = OtherState.EMPTY
    }

    fun postNetError() {
        otherState.value = OtherState.NET_ERROR
    }

    fun postServerError() {
        otherState.value = OtherState.SERVER_ERROR
    }

    fun postTokenError() {
        otherState.value = OtherState.TOKEN_ERROR
    }

    fun postError(errorCode: Int, msg: String?) {
        error.value = errorCode to msg
    }

}

await的代码如下:

inline fun <reified T> await(responseBean: () -> ResponseBean<T>): T {
    if (!"".isNetUsable)
        throw OperatorException(-2, "网络连接失败,请打开网络连接!")
    try {
        val response = responseBean.invoke()
        when (response.code) {
            0 -> {
                return response.data ?: T::class.java.newInstance()
            }
            2000 -> {
                ActivityTask.clearTask()
                Router.withApi(App::class.java).toLogin()
                throw OperatorException(2000, "您的账号在其它地方登陆,请保管好账号密码!")
            }
            else -> {
                if (response.msg?.contains(tokenError) == true) {
                    ActivityTask.clearTask()
                    Router.withApi(App::class.java).toLogin()
                    BaseApplication.instance.toast("您的账号在其它地方登陆,请保管好账号密码!")
                    throw OperatorException(2000, "您的账号在其它地方登陆,请保管好账号密码!")
                } else {
                    throw OperatorException(response.code, response.msg ?: "数据解析异常,请联系技术人员解决!")
                }
            }
        }
    } catch (e: OperatorException) {
        throw OperatorException(e.code, e.msg)
    } catch (e: SocketTimeoutException) {
        e.printStackTrace()
        throw OperatorException(-6, "网络连接超时,请稍后重试...")
    } catch (e: Exception) {
        e.printStackTrace()
        CrashReport.postCatchedException(e)
        throw OperatorException(-6, "网络连接异常!")
    }
}

4.第四步就是activity代码了,如下

mViewModel.mLoginLiveData.observes(this) {
            onStart {
                LoadingDialog.show(supportFragmentManager, "登录中")
            }
            onSuccess {
                PreferenceManager.user = it
                PreferenceManager.token = it.userToken!!
              
                //记录用户登录密码
                PreferenceManager.userLoginInfo = Pair(
                    username.text.toString(),
                    if (isRememberPassword) password.text.toString() else ""
                )
                startActivity<MainActivity>()
                finish()
            }
            onFailed { error, _ ->
                showTipToast(error.toString())
            }
            onNetFail {
                showTipToast("网络连接失败,请检查网络...")
            }
            onServerFail {
                showTipToast("服务器错误,请稍候重试")
            }
            onComplete {
                LoadingDialog.dismiss()
            }
        }

其中自定义扩展函数observes为:

inline fun <reified T> StateLiveData<T>.observes(
    owner: LifecycleOwner,
    dsl: RetrofitCoroutineDsl<T>.() -> Unit
) {
    val request = RetrofitCoroutineDsl<T>()
    request.dsl()
    observe(owner, Observer {
        //这块千万不要改,出错不负责
        request.onSuccess?.invoke(it ?: T::class.java.newInstance())
    })
    startState.observe(owner, Observer {
        request.onStart?.invoke()
    })
    otherState.observe(owner, Observer {
        when (it) {
            OtherState.NET_ERROR -> {
                request.onNetFail?.invoke()
            }
            OtherState.SERVER_ERROR -> {
                request.onServerFail?.invoke()
            }
            OtherState.EMPTY -> {
                d("collect observe onempty ${request.onEmpty == null}")
                request.onEmpty?.invoke()
            }
            OtherState.TOKEN_ERROR -> {
                ActivityTask.clearTask()
                Router.withApi(App::class.java).toLogin()
                BaseApplication.instance.toast("您的账号在其它地方登陆,请保管好账号密码!")
            }
            else -> {

            }
        }
    })
    completeState.observe(owner, Observer {
        request.onComplete?.invoke()
    })
    error.observe(owner, Observer {
        if (it?.second?.contains(tokenError) == true) {
            ActivityTask.clearTask()
            Router.withApi(App::class.java).toLogin()
            BaseApplication.instance.toast("您的账号在其它地方登陆,请保管好账号密码!")
        } else {
            request.onFailed?.invoke(it.second, it.first)
        }
    })
}

RetrofitCoroutineDsl代码为:

enum class OtherState {
    EMPTY, NET_ERROR, SERVER_ERROR, TOKEN_ERROR
}

class RetrofitCoroutineDsl<T> {
    lateinit var api: (ResponseBean<T>)
    var onSuccess: ((T) -> Unit)? = null
    var onEmpty: (() -> Unit)? = null
    var onComplete: (() -> Unit)? = null
    var onStart: (() -> Unit)? = null
    var onNetFail: (() -> Unit)? = null
    var onServerFail: (() -> Unit)? = null
    var onFailed: ((msg: String?, code: Int) -> Unit)? = null

    var showFailedMsg = false

    internal fun clean() {
        onSuccess = null
        onComplete = null
        onFailed = null
        onEmpty = null
        onStart = null
        onNetFail = null
        onServerFail = null
    }

    fun onSuccess(block: (T) -> Unit) {
        this.onSuccess = block
    }

    fun onComplete(block: () -> Unit) {
        this.onComplete = block
    }

    fun onEmpty(block: () -> Unit) {
        d("collect dsl onempty")
        this.onEmpty = block
    }

    fun onStart(block: () -> Unit) {
        this.onStart = block
    }

    fun onNetFail(block: () -> Unit) {
        this.onNetFail = block
    }

    fun onServerFail(block: () -> Unit) {
        this.onServerFail = block
    }

    fun onFailed(block: (error: String?, code: Int) -> Unit) {
        this.onFailed = block
    }

}

其中,采用kt的dsl写法,按需求,写你需要的方法
因此你的网络请求就很简单了,就这四步了,这个实践支持串行请求,只需要这样写:

fun login(userName:String,password:String){
        if (userName.isEmpty()){
            mLoginLiveData.postError(1,"用户名不能为空")
            return
        }
        if (password.isEmpty()){
            mLoginLiveData.postError(1,"密码不能为空")
            return
        }
        val map = hashMapOf<String, String>()
        map["adminLoginName"] = userName
        map["adminPassword"] = password
        launch(mLoginLiveData){
           val name = await { Repository.getName(xxx) }
           val password = await { Repository.getPassword(xxx) }
           await { Repository.login(name,password) }
        }
    }

可以看到网络请求思路清晰,且书写简单!!!

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