MVP设计模式

前言

以前在学习Android的时候,第一次接触设计模式,就是MVC。当时,写的代码不多,听得懵懵懂懂。在后来,深切感受到是在第一次组队写项目的时候,由于时间紧迫,前期我们就做了分工,我当时的任务就是搞定接口文档,并封装一个网络请求类,别的组员就暂时通过假数据做着自己的模块。

突然,我就感觉到其实 APP 很简单,就是UI+数据,没有真正数据的 APP 是死的,一旦有了数据与之互动,那么它就变活了,当然了,活得这么样又是另外一码事。那么,一手UI,一手数据,UI如何显示,数据什么时候更新,什么时候更新UI等等中间的互动就需要一个桥梁。

MVC

MVC, Model View Controller,这里的桥梁就是Controller,简单来说就是通过 Controller 的控制区操作 Model 层的数据,然后返回给 View 层显示。

mvc设计模式.png
  • Model : javaBean 业务逻辑相关
  • View : xml文件
  • Controller : activity fragment

在这里看 View 层,它是 xml 文件,它仅仅是一些布局文件,控制能力基本没有,如果想要去控制其中的控件隐藏/显示,只能将代码写到 activity 中,造成了 activity 既是 View 层也是 Controller 层的窘境。也就造成了 android中大部分app用的是view-model 。

Most of the modern Android applications just use View-Model architecture,everything is connected with Activity.

这就导致了,如果是一个逻辑很复杂的页面,activity/fragment的代码将相当的庞大臃肿,维护起来也不方便。

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_home);

    initRetrofit();
    initView();
    initEvent();
    initData();
}

private void initData() {
    getDatas(REFRESH_DATA);
}

private void initRetrofit() {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl("http://gank.io/")
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .build();
    mGankioService = retrofit.create(GankioService.class);
}

private void initEvent() {
    mSr.setOnRefreshListener(mOnRefreshListener);
}

private void initView() {
    mSr = (SwipeRecyclerView) findViewById(R.id.activity_swipe_recycler);
    /**设置布局管理器*/
    mLayoutManager = new GridLayoutManager(this, 3);
    mSr.setLayoutManager(mLayoutManager);
    /**设置adapter*/
    mMyadapter = new MyAdapter();
    mSr.setAdapter(mMyadapter);
}

class MyAdapter extends SwipeRecyclerView.Adapter<MyAdapter.MyViewHolder> {

    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        MyViewHolder vh = new MyViewHolder(LayoutInflater.from(HomeActivity.this).inflate(R.layout.activity_recycler_item, parent, false));
        return vh;
    }

    @Override
    public void onBindViewHolder(MyViewHolder holder, int position) {
        final String url = mDatas.get(position).url;

        Glide.with(holder.mImageView.getContext())
                .load(url)
                .centerCrop()
                .placeholder(R.color.app_primary_dark)
                .crossFade()
                .into(holder.mImageView);

        holder.mImageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(HomeActivity.this, BitmapActivity.class);
                intent.putExtra("url", url);
                startActivity(intent);
            }
        });
    }

    @Override
    public int getItemCount() {
        if (mDatas == null)
            return 0;
        return mDatas.size();
    }

    @Override
    protected void loadMoreDatas(RecyclerView recyclerView, int dx, int dy) {
        int position = mLayoutManager.findLastVisibleItemPosition();
        int itemCount = mMyadapter.getItemCount();
        if (position + 1 == itemCount) {
            getDatas(LOADMORE_DATA);
        }
    }

    public class MyViewHolder extends RecyclerView.ViewHolder {

        ImageView mImageView;

        public MyViewHolder(View itemView) {
            super(itemView);
            mImageView = (ImageView) itemView.findViewById(R.id.item_imageview);
        }
    }

}

private void getDatas(final int action) {

    if (action == REFRESH_DATA) {
        page = 1;
    } else {
        page++;
    }

    mGankioService.getWelfare(50, page)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(new Subscriber<BaseModel<List<Welfare>>>() {
                @Override
                public void onStart() {
                    mSr.mSwipeRefreshLayout.setRefreshing(true);
                }

                @Override
                public void onCompleted() {
                    mSr.mSwipeRefreshLayout.setRefreshing(false);
                }

                @Override
                public void onError(Throwable e) {
                    Log.i(">>>", "onError " + e.toString());
                }

                @Override
                public void onNext(BaseModel<List<Welfare>> listBaseModel) {
                    if (action == REFRESH_DATA) {
                        mDatas.clear();
                    }
                    mDatas.addAll(listBaseModel.results);
                    mMyadapter.notifyDataSetChanged();
                }
            });
}

private SwipeRecyclerView.OnRefreshListener mOnRefreshListener = new SwipeRecyclerView.OnRefreshListener() {
    @Override
    public void onRefresh() {
        getDatas(REFRESH_DATA);
    }
};

这是我之前用 mvc 写的gank.io其中一个显示妹子图的 activity。可以看到通过 retrofit 获取数据,并显示数据。

MVP

上文已经提到了 MVC 的缺陷,接着便演化出来了 MVP 。

mvp设计模式.png
  • Model : javaBean 业务逻辑相关
  • View : activity fragemnt
  • presenter : 负责完成View于Model间的交互

从图中可以看出, View 层和 Model 层已经完全的解耦了,以前的 Controller 换成了 Presenter 作为它们之间的桥梁,用于操作 View 层发出的事件传递到 Presenter 中,Presenter 层再去操作 Model 层,获取数据并返回给 View 层。

以下就将上文的代码利用 MVP 实现。

  1. Model

    在以上代码中,对数据进行的操作有,最开始的初始化数据,刷新数据,加载更多数据。对此创建一个 HomeModel

     public interface HomeModel {
         void initializeDatas();
         void refreshDatas();
         void loadMoreDatas();
     }
    

    实现之:

     public class HomeModelmp implements HomeModel {
         public static final int REFRESH_DATA = 1;
         public static final int LOADMORE_DATA = 2;
         private HomeOnListener mListener;
         private int page;
     
         public HomeModelmp(HomeOnListener l) {
             this.mListener = l;
         }
     
         @Override
         public void initializeDatas() {
             getDatas(REFRESH_DATA);
         }
     
         @Override
         public void refreshDatas() {
             getDatas(REFRESH_DATA);
         }
     
         @Override
         public void loadMoreDatas() {
             getDatas(LOADMORE_DATA);
         }
     
         private void getDatas(final int action) {
     
             if (action == REFRESH_DATA) {
                 page = 1;
             } else {
                 page++;
             }
     
             ServiceManager.getInstance().mService.getWelfare(20, page)
                     .subscribeOn(Schedulers.io())
                     .observeOn(AndroidSchedulers.mainThread())
                     .subscribe(new Subscriber<BaseModel<List<Welfare>>>() {
                         @Override
                         public void onStart() {
                             mListener.onStart();
                         }
     
                         @Override
                         public void onCompleted() {
                             mListener.onCompleted();
                         }
     
                         @Override
                         public void onError(Throwable e) {
                             mListener.onError(e);
                         }
     
                         @Override
                         public void onNext(BaseModel<List<Welfare>> listBaseModel) {
                             mListener.onNext(action, listBaseModel);
                         }
                     });
         }
     
         public interface HomeOnListener {
             void onStart();
     
             void onCompleted();
     
             void onError(Throwable e);
     
             void onNext(int action, BaseModel<List<Welfare>> listBaseModel);
         }
     }
    
  2. View

    View 层就是 activity ,其中将只有纯粹的UI操作。同样创建一个接口:

     public interface HomeView {
         /**
          * 加载中
          */
         void onLoading();
     
         /**
          * 加载完成
          */
         void onLoadCompleted();
     
         /**
          * 加载布局
          */
         View inflateLayout(ViewGroup parent, boolean b);
     
         /**
          * 跳转
          */
         void skipActivity(String url);
     
         /**
          * 获取最后一个可见item位置
          */
         int findLastPosition();
     }
    

    将activity实现该接口:

     public class HomeActivity extends AppCompatActivity implements HomeView {
     
         private SwipeRecyclerView mSr;
         private HomePresenter mHomePresenter;
         private GridLayoutManager mLayoutManager;
     
         @Override
         protected void onCreate(Bundle savedInstanceState) {
             super.onCreate(savedInstanceState);
             setContentView(R.layout.activity_home);
     
             if (mHomePresenter == null) {
                 mHomePresenter = new HomePresenter(this);
             }
             initView();
             initEvent();
             initData();
         }
     
         private void initData() {
             mHomePresenter.initialize();
         }
     
         private void initEvent() {
             mSr.setOnRefreshListener(mOnRefreshListener);
         }
     
         private void initView() {
             mSr = (SwipeRecyclerView) findViewById(R.id.activity_swipe_recycler);
             /**设置布局管理器*/
             mLayoutManager = new GridLayoutManager(this, 3);
             mSr.setLayoutManager(mLayoutManager);
             /**设置adapter*/
             mHomePresenter.setAdapter(mSr);
         }
     
         @Override
         public void onLoading() {
             mSr.mSwipeRefreshLayout.setRefreshing(true);
         }
     
         @Override
         public void onLoadCompleted() {
             mSr.mSwipeRefreshLayout.setRefreshing(false);
         }
     
         @Override
         public View inflateLayout(ViewGroup parent, boolean b) {
             return LayoutInflater.from(HomeActivity.this).inflate(R.layout.activity_recycler_item, parent, false);
         }
     
         @Override
         public void skipActivity(String url) {
             Intent intent = new Intent(HomeActivity.this, BitmapActivity.class);
             intent.putExtra("url", url);
             startActivity(intent);
         }
     
         @Override
         public int findLastPosition() {
             return mLayoutManager.findLastVisibleItemPosition();
         }
     
         private SwipeRecyclerView.OnRefreshListener mOnRefreshListener = new SwipeRecyclerView.OnRefreshListener() {
             @Override
             public void onRefresh() {
                 mHomePresenter.onRefreshData();
             }
         };
    
  3. Presenter

    Presenter作为 View 和 Model 之间交互的桥梁,也就是在合适的地方调用各自的方法。

     public class HomePresenter implements HomeModelmp.HomeOnListener {
     
         private HomeView mHomeView;
         private List<Welfare> mDatas = new ArrayList<>();
         private HomeActivity view;
         private HomeModel mHomeModel;
         private HomeAdapter mAdapter;
      
         public HomePresenter(HomeView homeView) {
             this.mHomeView = homeView;
             this.mHomeModel = new HomeModelmp(this);
         }
    
         public void initialize(){
             mDatas.clear();
             mHomeModel.initializeDatas();
         }
     
         public void onRefreshData() {
             mDatas.clear();
             mHomeModel.refreshDatas();
         }
     
         @Override
         public void onStart() {
             mHomeView.onLoading();
         }
     
         @Override
         public void onCompleted() {
             mHomeView.onLoadCompleted();
         }
     
         @Override
         public void onError(Throwable e) {
     
         }
     
         @Override
         public void onNext(int action, BaseModel<List<Welfare>> listBaseModel) {
             if (action == HomeModelmp.REFRESH_DATA) {
                 mDatas.clear();
             }
             mDatas.addAll(listBaseModel.results);
             mAdapter.notifyDataSetChanged();
         }
     
         public void setAdapter(SwipeRecyclerView sr) {
             mAdapter = new HomeAdapter();
             sr.setAdapter(mAdapter);
         }
     
         public class HomeAdapter extends SwipeRecyclerView.Adapter<HomeAdapter.MyViewHolder> {
     
             @Override
             public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
                 MyViewHolder vh = new MyViewHolder(mHomeView.inflateLayout(parent, false));
                 return vh;
             }
     
             @Override
             public void onBindViewHolder(MyViewHolder holder, int position) {
                 final String url = mDatas.get(position).url;
     
                 Glide.with(holder.mImageView.getContext())
                         .load(url)
                         .centerCrop()
                         .placeholder(R.color.app_primary_dark)
                         .crossFade()
                         .into(holder.mImageView);
     
                 holder.mImageView.setOnClickListener(new View.OnClickListener() {
                     @Override
                     public void onClick(View v) {
                         mHomeView.skipActivity(url);
                     }
                 });
             }
     
             @Override
             public int getItemCount() {
                 if (mDatas == null)
                     return 0;
                 return mDatas.size();
             }
     
             @Override
             protected void loadMoreDatas(RecyclerView recyclerView, int dx, int dy) {
                 int position = mHomeView.findLastPosition();
                 int itemCount = getItemCount();
                 if (position + 1 == itemCount) {
                     mHomeModel.loadMoreDatas();
                 }
             }
     
             public class MyViewHolder extends RecyclerView.ViewHolder {
     
                 public ImageView mImageView;
     
                 public MyViewHolder(View itemView) {
                     super(itemView);
                     mImageView = (ImageView) itemView.findViewById(R.id.item_imageview);
                 }
             }
     
         }
     }
    

Ok,大功告成。这里仅是贴出了关键代码,github地址:Gnak.iohttps://github.com/cgzysan/Gank.io</br>
参考资料:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容