Android搭建應(yīng)用框架系列之MVP封裝

前言

接著上一篇的博客,在上一篇中已經(jīng)把網(wǎng)絡(luò)框架Retrofit進(jìn)行簡(jiǎn)單的封裝徐矩,接下來說說MVP模式

MVC&MVP&MVVM區(qū)別

先來說說MVC抱完,如圖

MVC.png

圖是最好的記憶方式蔚舀,MVC呈現(xiàn)閉環(huán)的形式附较。其中:

  • View 使用xml文件做為視圖樣式
  • Controller Activity/Fragment控制業(yè)務(wù)邏輯吃粒,比如發(fā)起網(wǎng)絡(luò)請(qǐng)求,控制視圖里面的輸入輸出信息拒课,更新本地?cái)?shù)據(jù)庫(kù)信息等等
  • Model 網(wǎng)絡(luò)請(qǐng)求數(shù)據(jù)模型徐勃,本地?cái)?shù)據(jù)庫(kù)數(shù)據(jù)模型,以及一些耗時(shí)處理操作并更新View的展示信息等等

MVC具有兩種設(shè)計(jì)方式

  1. View獲取用戶操作信息早像,然后把信息傳遞給Controller進(jìn)行一些邏輯處理
  2. 直接把操作信息給Controller疏旨,比如Controller獲取EditText的輸入信息等等

MVC優(yōu)缺點(diǎn)

優(yōu)點(diǎn)

  • MVC已經(jīng)在framework層搭建好,上手比較容易扎酷,模塊化比較清晰
  • 實(shí)現(xiàn)了視圖層View,業(yè)務(wù)邏輯層Controller以及數(shù)據(jù)模型層Model的分離遏匆,降低了代碼的耦合性

缺點(diǎn)

  • Controller任務(wù)重法挨,容易導(dǎo)致代碼臃腫
  • View依賴Modellayout無(wú)法實(shí)現(xiàn)數(shù)據(jù)綁定幅聘,從而導(dǎo)致各種自定義 View凡纳,提高了代碼的重復(fù)性
  • 由于View依賴Model,導(dǎo)致不好寫代碼的測(cè)試用例帝蒿,只能編譯調(diào)試荐糜。

MVP

為了解決MVCView依賴Model,以及Controller代碼臃腫葛超,增加了Presenter來處理業(yè)務(wù)邏輯以及ViewModel的分離暴氏,如下圖

MVP.png

通過Presenter成功的把ViewModel進(jìn)行分離,其中:

  • ViewActivity/Fragment绣张,主要是實(shí)現(xiàn)MVPView的抽象方法供Presenter調(diào)用答渔,以及調(diào)用Presenter方法進(jìn)行網(wǎng)絡(luò)請(qǐng)求或者業(yè)務(wù)處理,也就是圖上的雙向通信
  • ModelMVC一樣侥涵,主要是網(wǎng)絡(luò)請(qǐng)求數(shù)據(jù)模型或者本地?cái)?shù)據(jù)庫(kù)數(shù)據(jù)模型以及一些耗時(shí)的操作沼撕。
  • Presenter 業(yè)務(wù)處理宋雏,調(diào)用Model的方法網(wǎng)絡(luò)請(qǐng)求獲取數(shù)據(jù)并且調(diào)用MVPView的方法更新View的數(shù)據(jù)

設(shè)計(jì)MVP的時(shí)候,會(huì)遇到一個(gè)問題务豺,就是在手機(jī)屏幕切換的時(shí)候磨总,如果保存Presenter當(dāng)前的狀態(tài)。這就涉及到Presenter生命周期問題笼沥。這個(gè)問題在網(wǎng)上能夠搜到很多資料蚪燕,主要解決辦法有:

  • Activity/Fragment當(dāng)做Presenter來使用,然后Presenter當(dāng)做View來使用敬拓,參考一種在android中實(shí)現(xiàn)MVP模式的新思路這篇文章
  • 使用Loader的生命周期來管理Presenter的生命周期
  • Presenter繼承Fragment來管理生命周期
    主要是以Presenter生命周期要比Activity/Fragment的生命周期更長(zhǎng)的原理來解決邻薯,這里使用的Loader類來解決。

MVP優(yōu)缺點(diǎn)
優(yōu)點(diǎn)

  • 實(shí)現(xiàn)了ViewModel的解耦
  • 便于寫代碼的測(cè)試用例

缺點(diǎn)

  • 上下文context容易丟失
  • 具有presenter生命周期問題以及內(nèi)存泄漏問題
  • 雙向通信回調(diào)乘凸,導(dǎo)致項(xiàng)目復(fù)雜化

MVVM
MVP中厕诡,當(dāng)我們?cè)?code>Presenter中獲取到Model需要調(diào)用MVPView的方法綁定View并更新數(shù)據(jù),而MVVM結(jié)合databinding可以直接響應(yīng)Model數(shù)據(jù)變化自動(dòng)化更新View营勤,使Activity/Fragment代碼變得更加簡(jiǎn)潔灵嫌。如下圖

MVVM.png
  • Model 同上,網(wǎng)絡(luò)數(shù)據(jù)模型或者本地?cái)?shù)據(jù)模型
  • Viewxml格式來表示視圖格式葛作,結(jié)合databinding可以直接綁定Model數(shù)據(jù)和視圖點(diǎn)擊事件
  • ViewModel 處理業(yè)務(wù)邏輯寿羞,發(fā)送網(wǎng)絡(luò)請(qǐng)求,更新Model數(shù)據(jù)等操作

主要步驟就是View獲取響應(yīng)信息后通知ViewModelModel更新數(shù)據(jù)赂蠢,數(shù)據(jù)更新后绪穆,Model通知ViewModel更新View展示的數(shù)據(jù)

MVVM優(yōu)缺點(diǎn)
優(yōu)點(diǎn)

  • ViewModel低耦合
  • Activity/Fragment代碼變得更加簡(jiǎn)潔,減少了更新View數(shù)據(jù)以及點(diǎn)擊事件的處理
  • 解決了MVPView(Activity/Fragment)Presenter相互持有的問題
  • 不在需要考慮主線程更新UI的問題

缺點(diǎn)

  • View層變得更加復(fù)雜
  • 數(shù)據(jù)綁定使得BUG不好調(diào)試以及更大的性能消耗

MVP實(shí)現(xiàn)

實(shí)現(xiàn)的幾個(gè)步驟:

  • MVPView虱岂,定義一些回調(diào)的方法玖院,并讓Activity/Fragment實(shí)現(xiàn)它
  • Presenter 獲取View實(shí)例,并提供幾個(gè)加載數(shù)據(jù)的加載框處理的回調(diào)方法
  • 使用Loader管理Presenter生命周期(一個(gè)PresenterFactory和一個(gè)PresenterLoader)
  • Activity/Fragment通過LoaderLoaderManager.LoaderCallbacks回掉獲取Presenter實(shí)例

先定義MVPView

object Mvp {
    interface View {
        val sub: CompositeDisposable
    }
}

其中CompositeDisposableBaseActivity實(shí)現(xiàn)創(chuàng)建CompositeDisposable對(duì)象第岖,統(tǒng)一管理接口請(qǐng)求的Obserable的綁定和解綁

定義Presenter

interface Presenter{
    fun attachView(attach: Any)
    fun detachView()
}

attachView() 綁定View难菌,也就是Activity/Fragment
detachView() 釋放View

通過BasePresenter實(shí)現(xiàn)如下

@Suppress("UNCHECKED_CAST")
abstract class BasePresenter<V : Mvp.View> : Presenter,IProgressDialog{

    private var attachView:V? = null
    private var dialog:IProgressDialog ?= null
    lateinit var mvpView:V
    override fun attachView(attach: Any) {
        this.attachView = attach as V
        this.dialog = attach as IProgressDialog
        mvpView = mvpView()
        initHandler()
    }

    override fun detachView() {
        this.attachView = null
    }

    private fun mvpView(): V {
        checkAttachView()
        return attachView!!
    }
    fun checkAttachView(){
        if(this.attachView==null) throw RuntimeException("Call the attachView method before using the Mvp.View")
    }

    open fun initHandler() {
    }

    override fun showLoading() {
        dialog?.showLoading()
    }

    override fun dismissLoading() {
        dialog?.dismissLoading()
    }

    override fun updateNextPage(haveNext: Boolean) {
        dialog?.updateNextPage(haveNext)
    }
}

attachView對(duì)應(yīng)View層的引用,attachActivity/Fragment
dialog 對(duì)應(yīng)BaseActivity實(shí)現(xiàn)的IProgressDialog接口蔑滓,實(shí)現(xiàn)showLoading()郊酒,dismissLoading(),updateNextPage()方法管理網(wǎng)絡(luò)請(qǐng)求彈出框顯示和消失的操作以及RecyclerView分頁(yè)加載的一些操作键袱。
mvpView供繼承BasePresenterPresenter網(wǎng)絡(luò)請(qǐng)求處理以及一些Activity/Fragment方法的調(diào)用或者MVPView的一些方法

Presenter生命周期管理

Loader類生命周期有:

onStartLoading()
ActivityonStart()調(diào)用之后燎窘,會(huì)回調(diào)這個(gè)方法創(chuàng)建一個(gè)Loader實(shí)例

onForceLoad()
當(dāng)onStartLoading調(diào)用forceLoad方法后,會(huì)回調(diào)這個(gè)方法蹄咖,可在這里創(chuàng)建Presenter荠耽。

deliverResult()
調(diào)用這個(gè)方法可以把Presenter分發(fā)給Activity/Fragment

onReset()
Loader銷毀前回調(diào)

定義一個(gè)創(chuàng)建Presenter的接口PresenterFactory

interface PresenterFactory<out P:BasePresenter<*>> {
        fun create() :P
}

這樣就可以在Activity/Fragment實(shí)現(xiàn)這個(gè)接口創(chuàng)建對(duì)應(yīng)Presenter。從而得到相應(yīng)Presenter的實(shí)例方法的引用比藻。
定義PresenterLoader實(shí)現(xiàn)Loader類并傳入PresenterFactory來管理生命周期

class PresenterLoader<P :BasePresenter<*>>(context: Context, val factory:PresenterFactory<P>): Loader<P>(context) {
    private var presenter:P? = null
    override fun onStartLoading() {
        if(presenter != null){
            deliverResult(presenter)
            return
        }
        forceLoad()
    }

    override fun onForceLoad() {
        presenter = factory.create()
        deliverResult(presenter)
    }
    override fun onReset() {
        presenter = null
    }
}

結(jié)合上面描述的Loader生命周期就可以很好的理解上面的代碼了铝量。

LoaderManager.LoaderCallbacks

最后一步在BaseLoaderActivity實(shí)現(xiàn)LoaderManager.LoaderCallbacks的回調(diào)方法

abstract class BaseLoaderActivity<P:BasePresenter<*>>: AppCompatActivity(),Mvp.View,LoaderManager.LoaderCallbacks<P>{
    protected var presenter:P? = null
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        supportLoaderManager.initLoader(Consts.BASE_ACTIVITY_LOADER_ID,null,this)
    }
    override fun onLoadFinished(loader: Loader<P>?, data: P) {
        presenter = data
        presenter?.attachView(this)
    }
    override fun onCreateLoader(id: Int, args: Bundle?): Loader<P>? {
        return createLoader()
    }

    override fun onLoaderReset(loader: Loader<P>?) {
        presenter = null
    }
    override fun onDestroy() {
        presenter?.detachView()
        super.onDestroy()
    }
    abstract fun createLoader():Loader<P>
}

Loader會(huì)先調(diào)用onCreateLoader創(chuàng)建一個(gè)Loader實(shí)例倘屹,然后調(diào)用onLoadFinished,我們可以在這里獲取到Presenter實(shí)例慢叨,然后當(dāng)Loader銷毀前調(diào)用onLoaderReset()纽匙,這里我們可以對(duì)Presenter進(jìn)行釋放。其中

supportLoaderManager.initLoader(Consts.BASE_ACTIVITY_LOADER_ID,null,this)

是進(jìn)行Loader初始化的一個(gè)操作
createLoader()供繼承BaseLoaderActivityActivity結(jié)合PresenterLoaderPresenterFactory創(chuàng)建Loader<P>對(duì)象拍谐。

同理烛缔,BaseLoaderFragment也如上創(chuàng)建即可。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末轩拨,一起剝皮案震驚了整個(gè)濱河市践瓷,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌亡蓉,老刑警劉巖晕翠,帶你破解...
    沈念sama閱讀 221,273評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異砍濒,居然都是意外死亡淋肾,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,349評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門爸邢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來樊卓,“玉大人,你說我怎么就攤上這事杠河÷刀” “怎么了?”我有些...
    開封第一講書人閱讀 167,709評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵券敌,是天一觀的道長(zhǎng)唾戚。 經(jīng)常有香客問我,道長(zhǎng)陪白,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,520評(píng)論 1 296
  • 正文 為了忘掉前任膳灶,我火速辦了婚禮咱士,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘轧钓。我一直安慰自己序厉,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,515評(píng)論 6 397
  • 文/花漫 我一把揭開白布毕箍。 她就那樣靜靜地躺著弛房,像睡著了一般。 火紅的嫁衣襯著肌膚如雪而柑。 梳的紋絲不亂的頭發(fā)上文捶,一...
    開封第一講書人閱讀 52,158評(píng)論 1 308
  • 那天荷逞,我揣著相機(jī)與錄音,去河邊找鬼粹排。 笑死种远,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的顽耳。 我是一名探鬼主播坠敷,決...
    沈念sama閱讀 40,755評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼射富!你這毒婦竟也來了膝迎?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,660評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤胰耗,失蹤者是張志新(化名)和其女友劉穎限次,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體宪郊,經(jīng)...
    沈念sama閱讀 46,203評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡掂恕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,287評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了弛槐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片懊亡。...
    茶點(diǎn)故事閱讀 40,427評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖乎串,靈堂內(nèi)的尸體忽然破棺而出店枣,到底是詐尸還是另有隱情,我是刑警寧澤叹誉,帶...
    沈念sama閱讀 36,122評(píng)論 5 349
  • 正文 年R本政府宣布鸯两,位于F島的核電站,受9級(jí)特大地震影響长豁,放射性物質(zhì)發(fā)生泄漏钧唐。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,801評(píng)論 3 333
  • 文/蒙蒙 一匠襟、第九天 我趴在偏房一處隱蔽的房頂上張望钝侠。 院中可真熱鬧,春花似錦酸舍、人聲如沸帅韧。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,272評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)忽舟。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間叮阅,已是汗流浹背刁品。 一陣腳步聲響...
    開封第一講書人閱讀 33,393評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留帘饶,地道東北人哑诊。 一個(gè)月前我還...
    沈念sama閱讀 48,808評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像及刻,于是被迫代替她去往敵國(guó)和親镀裤。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,440評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容