Android Mvp實踐

oden發表於2016-09-03

簡介

本文是參考google官方釋出的MVP架構demo以及前人對MVP實現方式的一些總結做的一個簡單demo,在這裡記錄一下一點心得,希望能給想用MVP的人一點幫助。 MVP框圖

總體框架

工程目錄結構

首先先看下整個工程的目錄結構: 目錄的程式碼組織方式是按照功能來組織的,功能內部分為xactivity、xcontract、xfragment、xpresenter四個類檔案(x代表業務名稱)。base資料夾存放一些公用的基類檔案,data資料夾存放業務邏輯相關的程式碼,utils資料夾則放一些公用的工具類。本demo實現的功能為:通過點選介面上的按鈕,獲取手機相關資訊,獲取過程中加入延時及等待提示(模擬網路),最終將資訊顯示於介面上(簡單演示,只是顯示了系統時間)。

基類

先看下BasePresenter與BaseView這兩個介面類,它們分別是所有Presenter與View的基類。

public interface BasePresenter {
    void start();
}複製程式碼

BasePresenter中含有方法start(),該方法的作用是presenter開始獲取資料並呼叫view中方法改變介面顯示,其呼叫時機是在Fragment類的onResume方法中。

public interface BaseView<T> {
    void setPresenter(T presenter);
}複製程式碼

BaseView中含方法setPresenter,該方法作用是將presenter例項傳入view中,其呼叫時機是在activity的presenter實現類的建構函式中。

契約類

public interface GetPhoneInfoContract {

    interface View extends BaseView<Presenter> {

        void setTime(String time);

        void showLoading();

        void hideLoading();
    }

    interface Presenter extends BasePresenter {

        void getTime();
    }

}複製程式碼

與比較常見的mvp實現不同,官方的實現中加入了契約類來統一管理view與presenter的所有的介面,這種方式使得view與presenter中有哪些功能,一目瞭然,維護起來也方便,該例項中presenter的介面實現獲取系統時間,view的介面實現時間的顯示以及提示對話方塊的顯示及隱藏。

MVP的組織方式

activity的作用

activity作為全域性的控制者,負責建立view以及presenter例項,並將二者聯絡起來,具體的view交由fragment來實現,兩者各司其職。

@EActivity(R.layout.get_phone_info_act)
public class GetPhoneInfoActivity extends ActionBarActivity {

    private FragmentManager fm;
    private GetPhoneInfoFragment mGetPhoneInfoFragment = new GetPhoneInfoFragment_();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setDefaultFragment();
        new GetPhoneInfoPresenter(mGetPhoneInfoFragment);
    }

    private void setDefaultFragment() {
        fm = getFragmentManager();
        FragmentTransaction transaction = fm.beginTransaction();
        transaction.add(R.id.fragcontent, mGetPhoneInfoFragment);
        transaction.commit();
    }

}複製程式碼

本例中,activity中通過setDefaultFragment()設定了fragment,之後例項化GetPhoneInfoPresenter,並將frament傳遞進去,實現在presenter中通過fragment的介面對view進行操作展示。

presenter的實現

public class GetPhoneInfoPresenter implements GetPhoneInfoContract.Presenter{

    private final GetPhoneInfoContract.View mGetPhoneInfoView;
    private PhoneInfoBiz phoneInfoBiz;

    public GetPhoneInfoPresenter(GetPhoneInfoContract.View getPhoneInfoView) {
        mGetPhoneInfoView = getPhoneInfoView;
        mGetPhoneInfoView.setPresenter(this);
        phoneInfoBiz = new PhoneInfoBizIml();
    }

    @Override
    public void start() {
        getTime();
    }

    @Override
    public void getTime() {
        mGetPhoneInfoView.showLoading();
        phoneInfoBiz.getPhoneInfo(new PhoneInfoBiz.GetPhoneInfoCallback() {
            @Override
            public void onGetPhoneInfo(PhoneInfo phoneInfo) {
                mGetPhoneInfoView.setTime(phoneInfo.getTime());
                mGetPhoneInfoView.hideLoading();
            }
        });
    }
}複製程式碼

presenter建構函式中呼叫了view的setPresenter方法將自身例項傳入,start方法中處理了資料載入與展示。如果需要介面做對應的變化,直接呼叫view層的方法即可,這樣view層與presenter層就能夠很好的被劃分。

view的實現

@EFragment(R.layout.get_phone_info_frag)
public class GetPhoneInfoFragment extends Fragment implements GetPhoneInfoContract.View {

    private GetPhoneInfoContract.Presenter mPresenter;
    ProgressDialog dialog;

    @ViewById
    TextView tv_time;

    @ViewById
    Button btn_get_time;

    @Click
    void btn_get_time() {
        mPresenter.getTime();
    }

    @AfterViews
    void initView() {
        dialog = new ProgressDialog(getActivity());
    }

    @Override
    public void onResume() {
        super.onResume();
        mPresenter.start();
    }

    @Override
    public void setPresenter(GetPhoneInfoContract.Presenter presenter) {
        if (presenter != null)
            mPresenter = presenter;
    }

    @Override
    @UiThread
    public void setTime(String time) {
        tv_time.setText(time);
    }

    @Override
    public void showLoading() {
        dialog.setTitle("請稍候");
        dialog.setMessage("loading!");
        dialog.show();
    }

    @Override
    public void hideLoading() {
        dialog.dismiss();
    }

}複製程式碼

setPresenter方法繼承於父類,通過該方法,view獲得了presenter得例項,從而可以呼叫presenter程式碼來處理業務邏輯。在onResume中還呼叫了presenter得start方法,處理資料的載入與展示。

簡單歸納

  • activity建立fragment及例項化presenter,在例項化的同時,將fragment作為引數傳遞進去,這樣一來在presenter中就可呼叫view的介面對介面進行更新、展示
  • presenter例項化時,還呼叫了view的setPresenter方法,將自身傳遞進去,這樣一來fragment獲得了presenter的例項,便可在view中呼叫presenter進行業務邏輯的操作
  • view及presenter擁有彼此的例項,實現了在view中呼叫presenter處理業務,處理完後再presenter中更新view。

model層

簡單介紹下model層,PhoneInfo物件儲存手機相關資訊,PhoneInfoBiz為藉口類實現該業務所需要的介面及回撥介面,PhoneInfoBizIml為介面的實現類。直接貼程式碼:

public interface PhoneInfoBiz {

    interface GetPhoneInfoCallback {

        void onGetPhoneInfo(PhoneInfo phoneInfo);
    }

    void getPhoneInfo(GetPhoneInfoCallback getPhoneInfoCallback);
}複製程式碼
public class PhoneInfoBizIml implements PhoneInfoBiz{

    @Override
    public void getPhoneInfo(final GetPhoneInfoCallback getPhoneInfoCallback) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    PhoneInfo phoneInfo = new PhoneInfo();
                    phoneInfo.setTime(System.currentTimeMillis() + "");
                    phoneInfo.setMobileType(Build.MODEL);
                    phoneInfo.setMobileVer(Build.VERSION.RELEASE);
                    Thread.sleep(1000);
                    getPhoneInfoCallback.onGetPhoneInfo(phoneInfo);
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }).start();
    }
}複製程式碼

總結

至此,一個簡單的mvp框架到此結束,對於mvp的使用目前也還在探索中,上例是結合官方釋出的demo做的一個簡化工程,有不足之處歡迎一起探討交流!

最後附上本文demo及官方demo的地址:

本文demo連結 官方demo連結

本文首發地址連結:www.codeceo.com/article/and…

相關文章