Android MVP 实践

  本文是参考google官方发布的MVP架构demo以及前人对MVP实现方式的一些总结做的一个简单demo,在这里记录一下一点心得,希望能给想用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 {

      void setPresenter(T presenter);

  }

  BaseView中含方法setPresenter,该方法作用是将presenter实例传入view中,其调用时机是在activity的presenter实现类的构造函数中。

  契约类

  public interface GetPhoneInfoContract {

      interface View extends BaseView {

          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方法,处理数据的加载与展示。

  简单归纳

  l activity创建fragment及实例化presenter,在实例化的同时,将fragment作为参数传递进去,这样一来在presenter中就可调用view的接口对界面进行更新、展示

  l presenter实例化时,还调用了view的setPresenter方法,将自身传递进去,这样一来fragment获得了presenter的实例,便可在view中调用presenter进行业务逻辑的操作

  l 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做的一个简化工程,有不足之处请谅解!

the end

评论(0)