帶你寫一個不一樣的MVP開發(fā)框架

與孤獨為伴铭腕,讓自己來一場涅槃

項目GitHub地址:https://github.com/ms-liu/ProjectFrameDemo

一券时、了解

在傳統(tǒng)的Android項目開發(fā)過程中,按照MVC模塊劃分的話黄选,往往會發(fā)現(xiàn)其實并不能很好的劃分出各自的職責(zé)出來蝇摸,Activity有時既需要扮演C的角色還需要扮演V的角色。這就導(dǎo)致Activity中代碼耦合現(xiàn)象嚴(yán)重办陷,尤其對于某些業(yè)務(wù)邏輯相對復(fù)雜的頁面貌夕,動不動就是上千行的代碼。
??可能對于這些頁面一代開發(fā)人員來說還能很愉快的接受民镜,但是當(dāng)新人來維護時啡专,這就會讓他很頭痛了;
??“他在寫些什么? (黑人問號 黑人問號)他應(yīng)該是個大神制圈,代碼我都看不懂们童,我要好好研究研究畔况!”
??在這種現(xiàn)象的基礎(chǔ)上,便有了MVP的開發(fā)模式:

  • Model:數(shù)據(jù)層——業(yè)務(wù)邏輯和實體類
  • View:視圖層——頁面展示
  • Presenter:邏輯層——數(shù)據(jù)和視圖層交互

單單從目前的分析來看慧库,MVP和MVC并沒有什么大的區(qū)別跷跪,當(dāng)然這兩者本身本質(zhì)上的區(qū)別不大,都是用作解耦V和M齐板,只是將MVC套到Android開發(fā)過程中的時候吵瞻,讓Activity等一些組件,角色扮演不是那么清晰甘磨,所以導(dǎo)致了問題產(chǎn)生橡羞。
??而在MVP的模式中,將Activity這些組件完全當(dāng)成View層宽档,讓Activity職責(zé)單一化尉姨。Presenter負(fù)責(zé)數(shù)據(jù)處理,然后通過接口的形式達(dá)到與View交互的目的吗冤。讓View和Model不在有交集又厉。


MVP結(jié)構(gòu)

??這篇文章不作MVP如何編寫的講解,如需了解可自行百度椎瘟,另外項目里面也包含了MVP 編寫代碼覆致。

二、提升與改進(重點與目的

其實有過了解或者使用MVP的開發(fā)人員肺蔚,應(yīng)當(dāng)都能體會感受到煌妈,MVP確實能夠?qū)ctivity中代碼簡化。但是對于那些業(yè)務(wù)復(fù)雜界面宣羊,Presenter中代碼也是會急劇增多璧诵,并且有時也會將View層中代碼放到Presenter中,重新回到老狀態(tài)仇冯。
??另外不知道有沒有人碰到和我一樣的問題之宿,就是在使用MVP的過程中,有時在打開多個Activity頁面后苛坚,回退過程中比被,在某些使用Fragment的Activity中會報出NullPointException,這是因為在我們打開過多頁面時泼舱,由于內(nèi)存和生命周期管理導(dǎo)致Fragment被系統(tǒng)回收等缀,但是我們并沒有將這些告知Presenter,從而發(fā)生NullPointException娇昙。
??下面我們就一起來解決尺迂,這些問題。

(一)框架分析
MVPP結(jié)構(gòu)

??在該框架中,我們在V和P之間加上了一個Proxy或Controller代理類枪狂,我們將會盡量少的讓P和V直接進行交互接觸危喉,而是通過Proxy與V進行交互宋渔,在Proxy中預(yù)先處理部分邏輯州疾,從而達(dá)到減輕Presenter職責(zé)的目的,讓Presenter中代碼更加簡潔皇拣,更加專注于業(yè)務(wù)處理邏輯严蓖。

(二)Code
代碼分包展示

1、Model層

  • 定義IModel數(shù)據(jù)接口
public interface IModel<T> {
    void setModel(T t);
    T getModel();
}
  • 實現(xiàn)IModel接口
public class ImproveModelImpl implements IModel<ImproveInfoBean>{
    private ImproveInfoBean mModel;
    @Override
    public void setModel(ImproveInfoBean improveInfoBean) {
        this.mModel = improveInfoBean;
    }

    @Override
    public ImproveInfoBean getModel() {
        if (this.mModel == null){
            this.mModel = new ImproveInfoBean();
        }
        return mModel;
    }
}

2氧急、View層代碼

  • 定義View生命周期監(jiān)聽接口
public interface OnViewStateListener {

    void onCreate();

    void onPause();

    void onResume();

    void onStop();

    void onDestroy();
}
  • 定義公共IView接口
public interface IView {
    // 綁定對View生命周期監(jiān)聽
    void bindListener(OnViewStateListener listener);

    Context getContext();

     // Toast提示
    void showToast(String message);

     //顯示加載對話框
    void showLoadingDialog(String message);

    // 隱藏加載對話框
    void hideLoadingDialog();
}
  • 編寫ViewDelegate
    對 View操作的委托類颗胡,實現(xiàn)IView和OnViewStateListener接口中的方法
public class ViewDelegate implements IView,OnViewStateListener {

    private Context mCtx;
    private ProgressDialog mProgressDialog;

    public ViewDelegate(Context context){
        this.mCtx = context;
    }

    private List<OnViewStateListener> mOnViewStateListeners ;

    @Override
    public void bindListener(OnViewStateListener listener) {
        //用數(shù)組管理每一個View的生命周期,避免一個頁面有多個監(jiān)聽
        if (mOnViewStateListeners == null){
            mOnViewStateListeners = new ArrayList<>();
            mOnViewStateListeners.add(listener);
        }else {
            if (!mOnViewStateListeners.contains(listener)){
                mOnViewStateListeners.add(listener);
            }
        }
    }

    @Override
    public Context getContext() {
        return mCtx;
    }

    @Override
    public void showToast(String message) {
        if (mCtx != null) {
            Toast.makeText(mCtx, message, Toast.LENGTH_SHORT).show();
        }
    }

    @Override
    public void showLoadingDialog(String message) {
        if (mCtx != null){
            mProgressDialog = new ProgressDialog(mCtx);
        }
    }

    @Override
    public void hideLoadingDialog() {
        if (mProgressDialog != null && mProgressDialog.isShowing()){
            mProgressDialog.hide();
        }
    }

    //------------------View生命周期管理--------------------------------------------------------

    @Override
    public void onCreate() {
        if (checkListener()){
            for (OnViewStateListener listener:
                 mOnViewStateListeners) {
                listener.onCreate();
            }
        }
    }

    @Override
    public void onPause() {
        if (checkListener()){
            for (OnViewStateListener listener:
                    mOnViewStateListeners) {
                listener.onPause();
            }
        }
    }

    @Override
    public void onResume() {
        if (checkListener()){
            for (OnViewStateListener listener:
                    mOnViewStateListeners) {
                listener.onResume();
            }
        }
    }

    @Override
    public void onStop() {
        if (checkListener()){
            for (OnViewStateListener listener:
                    mOnViewStateListeners) {
                listener.onStop();
            }
        }
    }

    @Override
    public void onDestroy() {
        if (checkListener()){
            for (OnViewStateListener listener:
                    mOnViewStateListeners) {
                listener.onDestroy();
            }
        }
    }

    private boolean checkListener(){
        return mOnViewStateListeners != null && !mOnViewStateListeners.isEmpty();
    }
}
  • 編寫B(tài)aseActivity吩坝,在里面實現(xiàn)頁面公共操作方法
public class BaseActivity extends AppCompatActivity implements IView{

    private ViewDelegate mDelegate;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        
        //可以考慮注解創(chuàng)建
        mDelegate = new ViewDelegate(this);

    }

    @Override
    public void bindListener(OnViewStateListener listener) {
        mDelegate.bindListener(listener);
    }

    @Override
    public Context getContext() {
        return BaseActivity.this;
    }

    @Override
    public void showToast(String message) {
        mDelegate.showToast(message);
    }

    @Override
    public void showLoadingDialog(String message) {
        mDelegate.showLoadingDialog(message);
    }

    @Override
    public void hideLoadingDialog() {
        mDelegate.hideLoadingDialog();
    }
}

3毒姨、Presenter層代碼

  • IPresenter公共接口,因為要監(jiān)聽View生命周期钉寝,讓它繼承OnViewStateListener 接口
public interface IImprovePresenter<M,V extends IView> extends OnViewStateListener {

    /**
     * View綁定
     * @param v
     */
    void bindView(V v);

    /**
     * 數(shù)據(jù)加載
     * @return
     */
    M loadModel();

    /**
     * View解綁
     */
    void detachView();
}
  • 編寫B(tài)asePresenter弧呐,實現(xiàn)部分共方法
public abstract class BaseImprovePresenter<M extends IModel,V extends IView> implements IImprovePresenter<M,V> {

    private List<String> mMethods;

    public BaseImprovePresenter(){
        this.mMethods = new ArrayList<>();
    }

    @Override
    public void onCreate() {

    }

    @Override
    public void onPause() {

    }

    @Override
    public void onResume() {

    }

    @Override
    public void onStop() {

    }

    @Override
    public void onDestroy() {

    }
    //---------------------模擬需要在生命周期中處理的邏輯--------------------------------------
    /**
     * 添加View生命周期結(jié)束時,需要結(jié)束的方法名稱;
     * 類似RxJava中添加addSubscription()
     * @param methodName
     */
    public void addHandleMethod(String methodName){
        if (mMethods != null){
            mMethods.add(methodName);
        }
    }

    /**
     * 清楚
     * 類似RxJava中添加clearSubscription()
     */
    public void clearMethod(){
        if (mMethods != null && !mMethods.isEmpty()){
            mMethods.clear();
        }
      }
}
  • 編寫Presenter嵌纲,
public class ImprovePresenter extends BaseImprovePresenter<ImproveModelImpl,IImproveView> {

    private IImproveView mView;
    private ImproveModelImpl mModel;

    @Override
    public void bindView(IImproveView iImproveView) {
        this.mView = iImproveView;
        mModel = loadModel();
        mView.showExplain(mModel.getModel().explain);

    }

    @Override
    public ImproveModelImpl loadModel() {
        addHandleMethod("異步請求數(shù)據(jù)方法");
        return new ImproveModelImpl();
    }

    @Override
    public void detachView() {
        clearMethod();
    }

    public void setInfo(String username, String password) {
        if (mModel != null){
            mModel.getModel().setName(username);
            mModel.getModel().setPassword(password);
        }
    }

    public void getInfoBean(){
        mView.showInfo("用戶名:"+mModel.getModel().getName()+"\r\n密碼:"+mModel.getModel().getPassword());
    }
}
  • 編寫PresenterProxy代理類
public class ImprovePresenterProxy implements IImprovePresenter<ImproveModelImpl,IImproveView>, View.OnClickListener {

    private ImprovePresenter mPresenter;
    private EditText etUserName;
    private EditText etPassword;
    private TextView componentShowInfo;
    private TextView componentExplain;
    private IImproveView mView;

    public ImprovePresenterProxy(ImprovePresenter improvePresenter){
        //判空
        checkPresenter(improvePresenter);
        //可以考慮依賴注入 方式
        this.mPresenter = improvePresenter;
        loadModel();
    }

    private void checkPresenter(ImprovePresenter improvePresenter) {
        if (improvePresenter == null){
            throw new NotBindPresenterException();
        }
    }

    @Override
    public ImproveModelImpl loadModel() {
        return mPresenter.loadModel();
    }

    @Override
    public void bindView(IImproveView iImproveView) {
        checkView(iImproveView);
        this.mView = iImproveView;
        mPresenter.bindView(iImproveView);
    }

    private void checkView(IImproveView iImproveView) {
        if (iImproveView == null){
            throw new NotBindViewException();
        }
    }

    @Override
    public void detachView() {
        mPresenter.detachView();
    }

    @Override
    public void onCreate() {
        //to do something
    }

    @Override
    public void onPause() {
        //to do something
    }

    @Override
    public void onResume() {
        //to do something
    }

    @Override
    public void onStop() {
        //to do something
    }

    @Override
    public void onDestroy() {
        detachView();
    }

    @Override
    public void onClick(View view) {
        int i = view.getId();
        if (i == R.id.btn_save) {
            if (etUserName != null && etPassword != null){
                mView.showToast("保存成功");
                mPresenter.setInfo(etUserName.getText().toString(),etPassword.getText().toString());
            }
        } else if (i == R.id.btn_get) {
            mPresenter.getInfoBean();
        } else {

        }
    }

    public void setComponentName(EditText etUsername) {
        this.etUserName = etUsername;
    }

    public void setComponentPassword(EditText etPassword) {
        this.etPassword = etPassword;
    }

    public void setComponentShowInfo(TextView componentShowInfo) {
        this.componentShowInfo = componentShowInfo;
    }

    public void setComponentExplain(TextView componentExplain) {
        this.componentExplain = componentExplain;
    }
}

至此俘枫,完成對MVP框架的完善和升級,完善后的MVP能夠更好的用于實際生產(chǎn)逮走,做到進一步的代碼解耦的目的鸠蚪。并且由于加入了對View生命周期的管理,也很好的解決NullPointException問題师溅。

實際代碼請下載或者Frok項目茅信。

項目GitHub地址:https://github.com/ms-liu/ProjectFrameDemo

歡迎大家給出中肯的建議和提高意見,大家一起學(xué)習(xí)進步墓臭。

個人郵箱:ms_liu163@163.com

QQ:275846421

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蘸鲸,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子起便,更是在濱河造成了極大的恐慌棚贾,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件榆综,死亡現(xiàn)場離奇詭異妙痹,居然都是意外死亡,警方通過查閱死者的電腦和手機鼻疮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進店門怯伊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人判沟,你說我怎么就攤上這事耿芹≌复郏” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵吧秕,是天一觀的道長琉闪。 經(jīng)常有香客問我,道長砸彬,這世上最難降的妖魔是什么颠毙? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮砂碉,結(jié)果婚禮上蛀蜜,老公的妹妹穿的比我還像新娘。我一直安慰自己增蹭,他們只是感情好滴某,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著滋迈,像睡著了一般霎奢。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上杀怠,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天椰憋,我揣著相機與錄音,去河邊找鬼赔退。 笑死橙依,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的硕旗。 我是一名探鬼主播窗骑,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼漆枚!你這毒婦竟也來了创译?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤墙基,失蹤者是張志新(化名)和其女友劉穎软族,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體残制,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡立砸,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了初茶。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片颗祝。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出螺戳,到底是詐尸還是另有隱情搁宾,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布倔幼,位于F島的核電站盖腿,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏凤藏。R本人自食惡果不足惜奸忽,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一堕伪、第九天 我趴在偏房一處隱蔽的房頂上張望揖庄。 院中可真熱鬧,春花似錦欠雌、人聲如沸蹄梢。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽禁炒。三九已至,卻和暖如春霍比,著一層夾襖步出監(jiān)牢的瞬間幕袱,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工悠瞬, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留们豌,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓浅妆,卻偏偏與公主長得像望迎,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子凌外,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

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