Android Clean 架構(gòu)淺析(基于JetPack AAC)

背景

本文基于Google官方推薦的AAC Samples 當(dāng)中的Use Cases示例對(duì)Clean架構(gòu)做一個(gè)簡(jiǎn)單的分析。
UserCase示例是用Kotlin編寫(xiě)的,使用了JetPack AAC當(dāng)中的部分組件:

  • ViewModel
  • LiveData
  • Data Binding
  • Navigation
  • Room

什么是Clean架構(gòu)

Clean 架構(gòu)一般指的代碼被劃分為多層少漆,類似于洋蔥的形狀僧诚,外層可以依賴內(nèi)層扯俱,內(nèi)層不能反向依賴外層脖律,內(nèi)層不知道外層的任何內(nèi)容。這里插一張官網(wǎng)Clean的分層結(jié)構(gòu)圖:

image

每個(gè)組件僅依賴于其下一級(jí)的組件蟀俊,例如Activity僅依賴ViewModel钦铺,ViewModel不能依賴任何視圖或者跟Activity上下文有關(guān)系的類。

為什么要增加UserCase肢预?

隨著業(yè)務(wù)的不斷擴(kuò)張矛洞,ViewModel的內(nèi)容可能會(huì)不斷膨脹,那么獨(dú)立出ViewModel的業(yè)務(wù)邏輯烫映,劃分到不同的領(lǐng)域(Use Cases)當(dāng)中是有必要的沼本,符合單一職責(zé)的指導(dǎo)思想,也有利于case的復(fù)用锭沟。體現(xiàn)到上圖中抽兆,就是在ViewModel和Reposity添加一個(gè)層級(jí),里面包含了不同的case族淮,ViewModel通過(guò)組合辫红、依賴注入的方式獲取Cases的能力凭涂。

代碼分析

以Task任務(wù)詳情界面為例,分析不同層級(jí)之間的協(xié)作方式。
View包括TaskDetailFragmenttaskdetail_frag.xml的內(nèi)容贴妻。TaskDetailFragment部分代碼如下:

class TaskDetailFragment : Fragment() {
    //Data Binding自動(dòng)生成的TaskdetailFragBinding類
    private lateinit var viewDataBinding: TaskdetailFragBinding
    //通過(guò)自定義的ViewModelFactory 工廠類創(chuàng)建TaskDetailViewModel對(duì)象
    private val viewModel by viewModels<TaskDetailViewModel> { getViewModelFactory() }
    ....
    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val view = inflater.inflate(R.layout.taskdetail_frag, container, false)
        viewDataBinding = TaskdetailFragBinding.bind(view).apply {
            viewmodel = viewModel
        }
        viewDataBinding.lifecycleOwner = this.viewLifecycleOwner
        viewModel.start(args.taskId)
        return view
    }
}

taskdetail_frag.xml:

<layout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
    <data>
        <import type="android.view.View" />
        <variable
            name="viewmodel"
            type="com.example.android.architecture.blueprints.todoapp.taskdetail.TaskDetailViewModel" />
    </data>

    <androidx.coordinatorlayout.widget.CoordinatorLayout
        android:id="@+id/coordinator_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <com.example.android.architecture.blueprints.todoapp.ScrollChildSwipeRefreshLayout
            android:id="@+id/refresh_layout"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:onRefreshListener="@{viewmodel::refresh}"
            app:refreshing="@{viewmodel.dataLoading}">
            ....
        </com.example.android.architecture.blueprints.todoapp.ScrollChildSwipeRefreshLayout>
    </androidx.coordinatorlayout.widget.CoordinatorLayout>
</layout>

通過(guò)Data Binding把視圖和ViewModel進(jìn)行了雙向綁定切油,數(shù)據(jù)是響應(yīng)式的,變化會(huì)實(shí)時(shí)反饋到界面上名惩,視圖不用再通過(guò)LiveData.observe()的方式對(duì)每個(gè)數(shù)據(jù)進(jìn)行監(jiān)聽(tīng)白翻,代碼更加簡(jiǎn)潔,ViewModel不依賴任何視圖绢片,也不會(huì)受configuration change影響。

ViewModel和UserCase交互

class TaskDetailViewModel(
    private val getTaskUseCase: GetTaskUseCase,
    private val deleteTaskUseCase: DeleteTaskUseCase,
    private val completeTaskUseCase: CompleteTaskUseCase,
    private val activateTaskUseCase: ActivateTaskUseCase
) : ViewModel() {
    val snackbarText: LiveData<Event<Int>> = _snackbarText
    private val _task = MutableLiveData<Task>()
    val task: LiveData<Task> = _task
    ...
    //設(shè)置task為完成狀態(tài)
    fun setCompleted(completed: Boolean) = viewModelScope.launch {
        val task = _task.value ?: return@launch
        if (completed) {
            completeTaskUseCase(task)
            showSnackbarMessage(R.string.task_marked_complete)
        } else {
            activateTaskUseCase(task)
            showSnackbarMessage(R.string.task_marked_active)
        }
    }
    //初始化task任務(wù)列表
    fun start(taskId: String?, forceRefresh: Boolean = false) {
        if (_isDataAvailable.value == true && !forceRefresh || _dataLoading.value == true) {
            return
        }
        // 展示loading框
        _dataLoading.value = true
        wrapEspressoIdlingResource {
        //viewModelScope內(nèi)的協(xié)程會(huì)在ViewModel銷毀后自動(dòng)取消
            viewModelScope.launch {
                if (taskId != null) {
                    //根據(jù)任務(wù)id獲取任務(wù)詳情
                    getTaskUseCase(taskId, false).let { result ->
                        if (result is Success) {
                            onTaskLoaded(result.data)
                        } else {
                            onDataNotAvailable(result)
                        }
                    }
                }
                //隱藏loading框
                _dataLoading.value = false
            }
        }
    }
    ....
    //更新toast提示信息
    private fun showSnackbarMessage(@StringRes message: Int) {
        _snackbarText.value = Event(message)
    }
}

UseCase最終會(huì)調(diào)用Repository的函數(shù)獲取數(shù)據(jù)岛琼,Repository是實(shí)現(xiàn)業(yè)務(wù)數(shù)據(jù)CURD的倉(cāng)庫(kù)底循,內(nèi)部數(shù)據(jù)可能從網(wǎng)絡(luò)獲取也可能來(lái)自本地緩存或者數(shù)據(jù)庫(kù),Repository通過(guò)接口的形式聲明槐瑞,內(nèi)部實(shí)現(xiàn)可以動(dòng)態(tài)替換熙涤,比如替換網(wǎng)絡(luò)庫(kù),更改數(shù)據(jù)存儲(chǔ)方式等困檩。

Clean架構(gòu)的優(yōu)勢(shì)

自此基本流程也分析完了祠挫,F(xiàn)ragment、ViewModel悼沿、UseCase和Repository各司其職等舔,互不依賴,對(duì)于提升代碼的復(fù)用性糟趾、可讀性慌植、穩(wěn)定性和可維護(hù)性都有很大的幫助。這樣分層對(duì)于測(cè)試也是友好的义郑,具體測(cè)試方式如下:

  • 展示層 (Presentation Layer) : 使用robolectric進(jìn)行集成和功能測(cè)試
  • 領(lǐng)域?qū)?(Domain Layer) :使用JUnit和Mockito進(jìn)行單元測(cè)試

參考資料

android-architecture-samples:https://github.com/android/architecture-samples/tree/usecases

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蝶柿,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子非驮,更是在濱河造成了極大的恐慌交汤,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,383評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件劫笙,死亡現(xiàn)場(chǎng)離奇詭異芙扎,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)邀摆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)纵顾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人栋盹,你說(shuō)我怎么就攤上這事施逾。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,852評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵汉额,是天一觀的道長(zhǎng)曹仗。 經(jīng)常有香客問(wèn)我,道長(zhǎng)蠕搜,這世上最難降的妖魔是什么怎茫? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,621評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮妓灌,結(jié)果婚禮上轨蛤,老公的妹妹穿的比我還像新娘。我一直安慰自己虫埂,他們只是感情好祥山,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著掉伏,像睡著了一般缝呕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上斧散,一...
    開(kāi)封第一講書(shū)人閱讀 49,929評(píng)論 1 290
  • 那天供常,我揣著相機(jī)與錄音,去河邊找鬼鸡捐。 笑死栈暇,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的闯参。 我是一名探鬼主播瞻鹏,決...
    沈念sama閱讀 39,076評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼鹿寨!你這毒婦竟也來(lái)了新博?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,803評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤脚草,失蹤者是張志新(化名)和其女友劉穎赫悄,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體馏慨,經(jīng)...
    沈念sama閱讀 44,265評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡埂淮,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了写隶。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片倔撞。...
    茶點(diǎn)故事閱讀 38,716評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖慕趴,靈堂內(nèi)的尸體忽然破棺而出痪蝇,到底是詐尸還是另有隱情鄙陡,我是刑警寧澤,帶...
    沈念sama閱讀 34,395評(píng)論 4 333
  • 正文 年R本政府宣布躏啰,位于F島的核電站趁矾,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏给僵。R本人自食惡果不足惜毫捣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評(píng)論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望帝际。 院中可真熱鬧蔓同,春花似錦、人聲如沸蹲诀。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,798評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)侧甫。三九已至,卻和暖如春蹋宦,著一層夾襖步出監(jiān)牢的瞬間披粟,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,027評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工冷冗, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留守屉,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,488評(píng)論 2 361
  • 正文 我出身青樓蒿辙,卻偏偏與公主長(zhǎng)得像拇泛,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子思灌,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評(píng)論 2 350