Android用MVP实现一个简单的类淘宝订单页面的示例

2020-10-27 21:02:34 浏览数 (1)

MVP(Model-View-Presenter) 是总所周知MVC模式的一个演变,他们的主要目的都是划分模块职责,降低模块耦合,易测试,提高代码复用,网上有很多相关的知识,这里仅是个人看法。

1、层级

Model:负责数据相关的操作

View:负责UI的绘制和用户的交互

Presenter:作为Model和View的中间协调部分,负责两者之间的业务逻辑处理

2、MVP的优缺点

优点:降低耦合,层级职责更明显,易于单元测试

缺点:造成类数量增多,在某些场景下presenter的复用会产生接口冗余

这几天把以前项目上做过的一个类似淘宝订单的页面单独提取出来重新做了一遍,把一些公共代码提取出来,方便以后用到类似界面简易集成,同时也用到了MVP框架,通过MVP之间各层次互相绑定的最基础的类,将数据和逻辑分离开,这是效果图

值得一提的是,这里用到了一些其他开源代码,比如刷新样式swipetoloadlayout等,具体的使用方法大家可以百度搜下名字就行了,资料很多。

4个Fragment用的统一的静态构建方法,配合ViewPager和TabLayout进行一次性集成

代码语言:javascript复制
//统一的Fragment构建方法
  public static MerchandiseListBaseFragment newInstance(int flag) {
    Bundle args = new Bundle();
    //type代表页签,0:全部订单 1:待发货 2:待收货 3:已完成
    args.putString("type", String.valueOf(flag));
    MerchandiseListBaseFragment fragment = new MerchandiseListBaseFragment();
    fragment.setArguments(args);
    return fragment;
  }
代码语言:javascript复制
  /**
   * 初始化ViewPager
   */
  private void initViewPager() {
    final ArrayList<Fragment  fragmentList = new ArrayList< ();
    MerchandiseListBaseFragment merchandiseListAll = MerchandiseListBaseFragment.
        newInstance(0);
    MerchandiseListBaseFragment merchandiseListDisPatch = MerchandiseListBaseFragment.
        newInstance(1);
    MerchandiseListBaseFragment merchandiseListReceive = MerchandiseListBaseFragment.
        newInstance(2);
    MerchandiseListBaseFragment merchandiseListFinish = MerchandiseListBaseFragment.
        newInstance(3);
    MerchandiseListBaseFragment.firstGetData();//初始化数据

    //持有fragment对象List
    fragmentList.add(merchandiseListAll);
    fragmentList.add(merchandiseListDisPatch);
    fragmentList.add(merchandiseListReceive);
    fragmentList.add(merchandiseListFinish);

    FragmentPagerAdapter fragmentAdapter = new BaseFragmentAdapter(getSupportFragmentManager(), fragmentList, mTitles);
    viewPager.setAdapter(fragmentAdapter);
    viewPager.setCurrentItem(0);
    viewPager.setOffscreenPageLimit(4);
    viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {

      @Override
      public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
      }

      @Override
      public void onPageSelected(int position) {
        tabLayout.getTabAt(position).select();
      }

      @Override
      public void onPageScrollStateChanged(int state) {
      }
    });
    tabLayout.setupWithViewPager(viewPager);//将tabLayout和ViewPager绑定
    tabLayout.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
      @Override
      public void onTabSelected(TabLayout.Tab tab) {
        viewPager.setCurrentItem(tab.getPosition());
      }

      @Override
      public void onTabUnselected(TabLayout.Tab tab) {
      }

      @Override
      public void onTabReselected(TabLayout.Tab tab) {
      }
    });
  }

订单列表是通过RecyclerView和swipetoloadlayout结合写的,具体代码参见源码(本文底部),因为这里只是提供一个淘宝订单页面的制作方法,所以这里的所有bean数据都暂时用的本地数据,没有添加网络框架进行数据加载,有兴趣的同学可以研究下MVP RxJava Okhttp Retrofit,非常好用。

MVP构建通过在view层和presenter层中的attachPre()方法进行对象绑定:

代码语言:javascript复制
  /**
   * MVP绑定
   */
  @Override
  public void attachPre() {
    activity = this;
    // mContext = this;
    mPresenter = TUtil.getT(this, ID_PRESENTER);
    mModel = TUtil.getT(this, ID_MODEL);
    if (mPresenter != null) {
      mPresenter.mContext = this;
    }
    initPresenter();
  }

这里的getT方法是用来得到类的模板参数的类型的,返回表示此类型实际类型参数的 Type 实例化对象。这样就可以获取view

代码语言:javascript复制
 public static <T  T getT(Object o, int i) {
    try {
      return ((Class<T ) ((ParameterizedType) (o.getClass()
          .getGenericSuperclass())).getActualTypeArguments()[i])
          .newInstance();
    } catch (InstantiationException e) {
      e.printStackTrace();
    } catch (IllegalAccessException e) {
      e.printStackTrace();
    } catch (ClassCastException e) {
      e.printStackTrace();
    }
    return null;
  }

源码地址:https://github.com/WeaponZhi/MerchandiseList

以后有机会再研究下网络框架和MVP的集成,希望对大家的学习有所帮助。

0 人点赞