Android中的MVP模式使用

jopen 8年前

         在讲MVP 之前,我们先来简单说下什么是MVC, 即Model(模型)、View(视图)、Control(控制器),相信大家对于MVC模式早已耳熟能详。原理性的东西这里不再多说。MVC在AndroidApp里面就有很好的体现。因为对于Android本身来说,界面部分的开发一般会用XML文件进行界面的描述开发。也就是MVC中的View层。而对于Model部分则大多是对应本地数据文件的读取或从网络获取数据。最后的Control控制器则有Activity来担当。MVC就说到这里,相信大多数Android猿们也是基于这种模式进行开发的。接下来,我们就来谈一谈今天的主角MVP模式在Android中又是如何实现的。

         何为MVP呢?即Model,VIew,Presenter(交互中间人相当于Control) 。但这里的Presenter隔离了Model和View之间的通信,使得Model和View之间能够完全解耦。在传统的MVC中,Activity担当了Control的角色,同时还要负责Dialog,Toast,PopupWindow的弹出,这就往往让Activity的责任变得繁重。一个Activity可能动不动就是几千行代码。而在MVP中View的角色不仅仅只是XML,而变成了Activity,Activity只负责View的工作。结合上面说的MVP能使Model和View之间能完全解耦,也就是说,在Activity中不会有任何关于Model层的操作,这就符合面向对象设计原则中的单一职责,让各个模块只负责自己的部分。更易于维护,代码也会更加简介。下面我会以一个小的项目来演示MVP在Android中如何使用。我在今日头天的web页拿到了一个新闻的URL,获取Json数据,解析显示

Bean

package himan.mvp.bean;    import java.io.Serializable;    public class NewsInfo implements Serializable {     /**    *     */   private static final long serialVersionUID = 1L;   private String datetime;   private int id;   private String title;   private String imageUrl;   private String abstractInfo;     public String getDatetime() {    return datetime;   }     public void setDatetime(String datetime) {    this.datetime = datetime;   }     public int getId() {    return id;   }     public void setId(int id) {    this.id = id;   }     public String getTitle() {    return title;   }     public void setTitle(String title) {    this.title = title;   }     public String getImageUrl() {    return imageUrl;   }     public void setImageUrl(String imageUrl) {    this.imageUrl = imageUrl;   }     public String getAbstractInfo() {    return abstractInfo;   }     public void setAbstractInfo(String abstractInfo) {    this.abstractInfo = abstractInfo;   }     public NewsInfo(String datetime, int id, String title, String imageUrl,     String abstractInfo) {    super();    this.datetime = datetime;    this.id = id;    this.title = title;    this.imageUrl = imageUrl;    this.abstractInfo = abstractInfo;   }     public NewsInfo() {   }    }

Model层的接口:还记得前两篇帖子讲的面向对象的六大设计模式精髓:面向接口编程,依赖于抽象

package himan.mvp.model;    import himan.mvp.bean.NewsInfo;    import java.util.List;    public interface INewsModel {     /**    * 加载数据    *     * @param dataListener    */   void loadNews(IOnDataListener<List<NewsInfo>> dataListener);     /**    * 缓存数据    *     * @param listNews    */   void saveNews(List<NewsInfo> listNews);  }

Model实现类

package himan.mvp.modelimpl;    import java.util.List;    import himan.mvp.bean.NewsInfo;  import himan.mvp.model.INewsModel;  import himan.mvp.model.IOnDataListener;  import himan.mvp.net.NewsNetHelper;  import himan.mvp.net.volley.UIDataListener;    public class NewsModelImpl implements INewsModel {     @Override   public void loadNews(final IOnDataListener<List<NewsInfo>> dataListener) {  // 回调接口    NewsNetHelper netHelper = new NewsNetHelper();    netHelper.setDataListener(new UIDataListener<List<NewsInfo>>() {       @Override     public void onDataChanged(List<NewsInfo> data) {      if (data == null) {       dataListener.error();      } else {       dataListener.completeLoad(data);      }     }    });    netHelper      .doHttpGet("http://toutiao.com/api/article/recent/?source=2&count=20&category=__all__&max_behot_time=1449467901.41&utm_source=toutiao&offset=0&_=1449468004256");   }     @Override   public void saveNews(List<NewsInfo> listNews) {    // 缓存数据到本地   }    }

View层接口:依赖于抽象编程

package himan.mvp.view;    import himan.mvp.bean.NewsInfo;    import java.util.List;    public interface INewsView {   /**    * 显示新闻列表    *     * @param listNews    */   public void showNewsList(List<NewsInfo> listNews);     /**    * 显示正在加载中进度条    */   public void showLoading();     /**    * 关闭正在加载中进度条    */   public void hideLoading();      /**    * 显示数据加载失败    */   public void showError();     }

MVP关键之处:中间人Presenter

package himan.mvp.presenter;    import java.util.List;    import himan.mvp.bean.NewsInfo;  import himan.mvp.model.INewsModel;  import himan.mvp.model.IOnDataListener;  import himan.mvp.modelimpl.NewsModelImpl;  import himan.mvp.view.INewsView;    public class NewsPresenter {           // 同时持有Model层和View层的引用   private INewsView mNewsView;   private INewsModel mNewsModel = new NewsModelImpl();     public NewsPresenter(INewsView newsView) {    this.mNewsView = newsView;   }   public void showNews() {    mNewsView.showLoading();    mNewsModel.loadNews(new IOnDataListener<List<NewsInfo>>() {       @Override     public void error() {      // 加载失败      mNewsView.hideLoading();      mNewsView.showError();     }       @Override     public void completeLoad(List<NewsInfo> t) {      mNewsView.hideLoading();      mNewsView.showNewsList(t);     }    });   }    }

Activity实现了View接口

package himan.mvp;    import himan.mvp.adapter.NewsAdapter;  import himan.mvp.bean.NewsInfo;  import himan.mvp.net.volley.VolleyQueueController;  import himan.mvp.presenter.NewsPresenter;  import himan.mvp.utils.LoadingWindow;  import himan.mvp.view.INewsView;    import java.util.ArrayList;  import java.util.List;    import android.app.Activity;  import android.os.Bundle;  import android.widget.ListView;  import android.widget.Toast;    import com.nostra13.universalimageloader.core.ImageLoader;  import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;    /**   * 测试 加载的数据 是从今日头条官网的直接push下来的   *    * @author Mr.Himan   *    */  public class MainActivity extends Activity implements INewsView {     private ListView mlVNews;     private NewsAdapter mNewsAdapter;     private List<NewsInfo> mListNews;     private NewsPresenter mNewsPresenter;     @Override   protected void onCreate(Bundle savedInstanceState) {    super.onCreate(savedInstanceState);    setContentView(R.layout.activity_main);    // 初始化Volley框架    VolleyQueueController.init(this);    // ImageLoader 默认配置参数    ImageLoaderConfiguration configuration = ImageLoaderConfiguration      .createDefault(this);    // 初始化ImageLoader    ImageLoader.getInstance().init(configuration);    initView();    mNewsPresenter = new NewsPresenter(this);    mNewsPresenter.showNews();     }     private void initView() {    mlVNews = (ListView) findViewById(R.id.news_lv_news_list);    mListNews = new ArrayList<NewsInfo>();    mNewsAdapter = new NewsAdapter(this, mListNews, R.layout.lv_item_news);    mlVNews.setAdapter(mNewsAdapter);   }     @Override   public void showNewsList(List<NewsInfo> listNews) {    mListNews.addAll(listNews);    mNewsAdapter.notifyDataSetChanged();   }     @Override   public void showLoading() {    Toast.makeText(this, "正在加载数据...", Toast.LENGTH_LONG).show();   }     @Override   public void hideLoading() {    Toast.makeText(this, "数据加载成功", Toast.LENGTH_SHORT).show();   }     @Override   public void showError() {     }    }

这就是MVP实现一个新闻展示的全部代码(貌似头条的接口关掉了 - -),Activity实现了View接口,而Presenter中保存有View和Model层的引用,使得Presenter能通过Model拿到数据,通过View的引用控制视图的显示。流程大概是这样,Presenter先通过Model的引用拿到数据,然后通过View的引用进行数据的显示,视图更新。Activity中的代码也非常简介。通过上面代码我们能够看出MVP的优点还是很多的,首先使得Model和View层完全解耦,我们知道项目中改动最多的一般都是View层,这就让程序易于维护,而且能够高度复用Presenter,而且也易于后期的拓展。缺点也显而易见,就是爆炸式增长的类和接口。复用的时候也可能造成接口的冗余。最大的问题是Presenter持有着View的强应用,在请求网络数据等耗时操作的时候,Activity可能被销毁,可能会导致View无法回收,而造成内存问题。关于这个问题,我会在下一篇MVP的帖子中讲解如何处理,欢迎关注





        

来自: http://blog.csdn.net/soul_code/article/details/50207857