搭建屬于自己的Android MVP 框架

本文主要是結(jié)合自己對MVP的理解搭建了符合自身業(yè)務(wù)場景的MVP框架。

先放一個(gè)Demo地址,文章末尾也有

關(guān)于MVP

  • M(Model)負(fù)責(zé)數(shù)據(jù)的請求准谚,解析谦炒,過濾等數(shù)據(jù)操作贯莺。
  • V(View)負(fù)責(zé)處理UI,通常以Activity Fragment的形式出現(xiàn)宁改。
  • P(Presenter)View Model中間件缕探,交互的橋梁。
    圖片.png

    上圖引用自此

MVP的好處

  • 分離了UI邏輯和業(yè)務(wù)邏輯还蹲,降低了耦合爹耗。
  • Activity只處理UI相關(guān)操作,代碼變得更加簡潔谜喊。
  • UI邏輯和業(yè)務(wù)邏輯抽象到接口中鲸沮,方便閱讀及維護(hù)。
  • 把業(yè)務(wù)邏輯抽到Presenter中去锅论,避免復(fù)雜業(yè)務(wù)邏輯造成的內(nèi)存泄漏讼溺。

具體實(shí)現(xiàn)

1.對View進(jìn)行封裝
一般情況下,做數(shù)據(jù)請求都有顯示加載框最易、請求成功怒坯、請求失敗等操作,我們把這些共有的功能封裝到BaseView中藻懒。


public interface IBaseView {

    /**
     * 顯示加載框
     */
    void showLoading();

    /**
     * 隱藏加載框
     */
    void dismissLoading();

    /**
     * 空數(shù)據(jù)
     *
     * @param tag TAG
     */
    void onEmpty(Object tag);

    /**
     * 錯(cuò)誤數(shù)據(jù)
     *
     * @param tag      TAG
     * @param errorMsg 錯(cuò)誤信息
     */
    void onError(Object tag, String errorMsg);

    /**
     * 上下文
     *
     * @return context
     */
    Context getContext();
}

2.對Presenter封裝
為了避免持有View的Presenter做耗時(shí)操作而引起的內(nèi)存泄漏剔猿,我們的Presenter應(yīng)該和宿主Activity/Fragment同創(chuàng)建、同銷毀嬉荆。

public abstract class BasePresenter{
    ...
    /**
     * 綁定View
     */
    public void attachView(View view) {
        this.view=view;
    }
    /**
     * 解綁View
     */
    public void detachView() {
         this.view=null;
    }
  ...
}

public abstract class MvpActivity extends BaseActivity implements View{
    ...
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //創(chuàng)建present
        presenter = createPresenter();
        if (presenter != null) {
            presenter.attachView(this);
        }
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (presenter != null) {
            presenter.detachView();
            presenter = null;
        }
    }
    ...
}

如上操作固然可以解決內(nèi)存泄漏問題归敬,但又會(huì)引發(fā)行的問題:

場景:用戶打開商品列表頁,網(wǎng)絡(luò)不好獲取數(shù)據(jù)比較慢,用戶離開該頁面汪茧,繼續(xù)瀏覽其他頁面椅亚,突然應(yīng)用崩潰了。

分析問題:
在用戶打開頁面的時(shí)候綁定P和V舱污,離開頁面的時(shí)候解綁P和V呀舔,當(dāng)耗時(shí)操作完成調(diào)用V更新界面,此時(shí)由于P和V已經(jīng)解綁V處于null扩灯,調(diào)用V的更新頁面方法就會(huì)引起空指針異常媚赖。

解決問題:
使用動(dòng)態(tài)代理對View做弱引用蔚万,完整的BasePresenter如下:


public abstract class BasePresenter<M extends IBaseModel, V extends IBaseView> {

    private V mProxyView;
    private M module;
    private WeakReference<V> weakReference;

    /**
     * 綁定View
     */
    @SuppressWarnings("unchecked")
    public void attachView(V view) {
        weakReference = new WeakReference<>(view);
        mProxyView = (V) Proxy.newProxyInstance(
                view.getClass().getClassLoader(),
                view.getClass().getInterfaces(),
                new MvpViewHandler(weakReference.get()));
        if (this.module == null) {
            this.module = createModule();
        }
    }

    /**
     * 解綁View
     */
    public void detachView() {
        this.module = null;
        if (isViewAttached()) {
            weakReference.clear();
            weakReference = null;
        }
    }

    /**
     * 是否與View建立連接
     */
    protected boolean isViewAttached() {
        return weakReference != null && weakReference.get() != null;
    }

    protected V getView() {
        return mProxyView;
    }

    protected M getModule() {
        return module;
    }

    protected Context getContext() {
        return getView().getContext();
    }

    protected void showLoading() {
        getView().showLoading();
    }

    protected void dismissLoading() {
        getView().dismissLoading();
    }


    /**
     * 通過該方法創(chuàng)建Module
     */
    protected abstract M createModule();

    /**
     * 初始化方法
     */
    public abstract void start();


    /**
     * View代理類  防止 頁面關(guān)閉P異步操作調(diào)用V 方法 空指針問題
     */
    private class MvpViewHandler implements InvocationHandler {

        private IBaseView mvpView;

        MvpViewHandler(IBaseView mvpView) {
            this.mvpView = mvpView;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            //如果V層沒被銷毀, 執(zhí)行V層的方法.
            if (isViewAttached()) {
                return method.invoke(mvpView, args);
            } //P層不需要關(guān)注V層的返回值
            return null;
        }
    }
}

3.契約類Contract的出現(xiàn)
通過契約類來管理Model鳞疲、View、Presenter的所有接口赏寇,這樣使得Presenter和View有哪些功能一目了然捻撑,維護(hù)起來也方便磨隘,同時(shí)使得View與Presenter一一對應(yīng),并有效地減少類的數(shù)目布讹。

public interface  Contract {

    interface Model extends IBaseModel {
        void login(User user, ResponseCallback callback);
    }

    interface View extends IBaseView {
        User getUserInfo();
        void loginSuccess(User user);
    }

    interface Presenter {
        void login();
    }
}

4.對Activity的封裝琳拭,Fragment封裝同理

public abstract class BaseMvpActivity<P extends BasePresenter> extends Activity implements IBaseView {

    protected P presenter;

    @SuppressWarnings("unchecked")
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //創(chuàng)建present
        presenter = createPresenter();
        if (presenter != null) {
            presenter.attachView(this);
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (presenter != null) {
            presenter.detachView();
            presenter = null;
        }

    }
    @Override
    public void showLoading() {
        if (loadingDialog != null && !loadingDialog.isShowing()) {
            loadingDialog.show();
        }
    }

    @Override
    public void dismissLoading() {
        if (loadingDialog != null && loadingDialog.isShowing()) {
            loadingDialog.dismiss();
        }
    }

    @Override
    public void onEmpty(Object tag) {

    }

    @Override
    public void onError(Object tag, String errorMsg) {

    }

    @Override
    public Context getContext() {
        return mContext;
    }
    /**
     * 創(chuàng)建Presenter
     */
    protected abstract P createPresenter();
}

通過泛型規(guī)定Presenter,并且暴露抽象方法createPresenter()給子類來創(chuàng)建Presenter描验,基類實(shí)現(xiàn)BaseView中的公共方法白嘁,減少子類代碼的冗余。
5.登錄案例


類結(jié)構(gòu)圖.jpg

契約類

public interface LoginContract {

    interface Model extends IBaseModel {

        /**
         * 登錄
         *
         * @param user     用戶信息
         * @param callback 回調(diào)
         */
        void login(User user, ResponseCallback callback);
    }

    interface View extends IBaseView {


        /**
         * 返回用戶信息
         */
        User getUserInfo();

        /**
         * 登錄成功
         */
        void loginSuccess(User user);

    }

    interface Presenter {

        /**
         * 登錄
         */
        void login();
    }
}

Model

public class LoginModel implements LoginContract.Model {

    @Override
    public void login(User user, ResponseCallback callback) {
        if (user == null) {
            callback.onError("", (Throwable) new Exception("用戶信息為空"));
        }
        RequestParam param = new RequestParam();
        param.addParameter("username", user.getUsername());
        param.addParameter("password", user.getPassword());
        HttpUtils.getInstance()
                .postRequest(Api.LOGIN, param, callback);
    }
}

Presenter

public class LoginPresenter extends BasePresenter<LoginContract.Model, LoginContract.View>
        implements LoginContract.Presenter {

    @Override
    public void login() {
        if (isViewAttached()) {
            getView().showLoading();
            getModule().login(getView().getUserInfo(), new OnResultObjectCallBack<User>() {
                @Override
                public void onSuccess(boolean success, int code, String msg, Object tag, User response) {
                    if (code == 0 && response != null) {
                        getView().loginSuccess(response);
                    } else {
                        getView().onError(tag, msg);
                    }
                }

                @Override
                public void onFailure(Object tag, Exception e) {
                    getView().onError(tag, msg);
                }

                @Override
                public void onCompleted() {
                    getView().dismissLoading();
                }
            });
        }
    }


    @Override
    protected LoginModel createModule() {
        return new LoginModel();
    }

    @Override
    public void start() { }
}

登錄Activity

public class LoginActivity extends ActionBarActivity<LoginPresenter> implements LoginContract.View {

    @BindView(R2.id.edt_name)
    EditText edtName;

    @BindView(R2.id.edt_pwd)
    EditText edtPwd;

    @BindView(R2.id.ob_login)
    ObserverButton obLogin;

    @BindView(R2.id.ob_register)
    TextView obRegister;

    @Override
    protected int getLayoutId() {
        return R.layout.user_activity_login;
    }

    @Override
    protected void initView() {
        setTitleText("登錄");
        obLogin.observer(edtName, edtPwd);
    }


    @OnClick({R2.id.ob_login, R2.id.ob_register})
    public void onViewClicked(View view) {
        int i = view.getId();
        if (i == R.id.ob_login) {
            presenter.login();
        } else if (i == R.id.ob_register) {
            ActivityToActivity.toActivity(mContext, RegisterActivity.class);
        }
    }

    @Override
    public void loginSuccess(User user) {
        UserInfoUtils.saveUser(user);
        EventBusUtils.sendEvent(new Event(EventAction.EVENT_LOGIN_SUCCESS));
        finish();
    }


    @Override
    public void onError(Object tag, String errorMsg) {
        super.onError(tag, errorMsg);
        ToastUtils.showToast(mContext, errorMsg);
    }

    @Override
    protected LoginPresenter createPresenter() {
        return new LoginPresenter();
    }

    @Override
    public void onEventBus(Event event) {
        super.onEventBus(event);
        if (TextUtils.equals(event.getAction(), EventAction.EVENT_REGISTER_SUCCESS)) {
            finish();
        }
    }

    @Override
    protected boolean regEvent() {
        return true;
    }

    @Override
    public User getUserInfo() {
        return new User(edtName.getText().toString().trim(), edtPwd.getText().toString().trim());
    }
}

總結(jié)

無論是MVP還是MCV或者M(jìn)VVM膘流,都是為把業(yè)務(wù)與UI分離絮缅,避免在一個(gè)Activity里把所有的操作都塞進(jìn)來,各自在各自的領(lǐng)域工作呼股。每個(gè)人對于層級結(jié)構(gòu)都有不同的理解和看法耕魄,封裝一個(gè)適合自己、適合當(dāng)下業(yè)務(wù)場景的框架才是最重要的彭谁。

這里的框架中所使用的就是MVP結(jié)構(gòu)

最后放上Demo地址吸奴,共同學(xué)習(xí),有什么不好的地方缠局,歡迎大家指出则奥!

參考文獻(xiàn)
Google爸爸的案例
JesseBraveMan的 Android MVP架構(gòu)搭建
淺談Android中的MVP架構(gòu)
深入講解Android MVP框架

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市狭园,隨后出現(xiàn)的幾起案子读处,更是在濱河造成了極大的恐慌,老刑警劉巖唱矛,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件罚舱,死亡現(xiàn)場離奇詭異井辜,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)管闷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門粥脚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人渐北,你說我怎么就攤上這事阿逃∶。” “怎么了赃蛛?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長搀菩。 經(jīng)常有香客問我呕臂,道長,這世上最難降的妖魔是什么肪跋? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任歧蒋,我火速辦了婚禮,結(jié)果婚禮上州既,老公的妹妹穿的比我還像新娘谜洽。我一直安慰自己,他們只是感情好吴叶,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布阐虚。 她就那樣靜靜地躺著,像睡著了一般蚌卤。 火紅的嫁衣襯著肌膚如雪实束。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天逊彭,我揣著相機(jī)與錄音咸灿,去河邊找鬼。 笑死侮叮,一個(gè)胖子當(dāng)著我的面吹牛避矢,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播囊榜,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼审胸,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了锦聊?” 一聲冷哼從身側(cè)響起歹嘹,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎孔庭,沒想到半個(gè)月后尺上,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體材蛛,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年怎抛,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了卑吭。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡马绝,死狀恐怖豆赏,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情富稻,我是刑警寧澤掷邦,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站椭赋,受9級特大地震影響抚岗,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜哪怔,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一宣蔚、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧认境,春花似錦胚委、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至茉盏,卻和暖如春鉴未,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背鸠姨。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工铜秆, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人讶迁。 一個(gè)月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓连茧,卻偏偏與公主長得像,于是被迫代替她去往敵國和親巍糯。 傳聞我的和親對象是個(gè)殘疾皇子啸驯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

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