Android中的MVP模式使用

时间:2022-10-12 21:15:09

         在讲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的帖子中讲解如何处理,欢迎关注