综述
对于MVP (Model View Presenter)架构是从著名的MVC(Model View Controller)架构演变而来的。而对于Android应用的开发中本身可视为一种MVC架构。通常在开发中将XML文件视为MVC中的View角色,而将Activity则视为MVC中的Controller角色。不过更多情况下在实际应用开发中Activity不能够完全充当Controller,而是Controller和View的合体。于是Activity既要负责视图的显示,又要负责对业务逻辑的处理。这样在Activity中代码达到上千行,甚至几千行都不足为其,同时这样的Activity也显得臃肿不堪。所以对于MVC架构并不很合适运用于Android的开发中
MVP架构简介
对于一个应用而言我们需要对它抽象出各个层面,而在MVP架构中它将UI界面和数据进行隔离,所以我们的应用也就分为三个层次。
- View: 对于View层也是视图层,在View层中只负责对数据的展示,提供友好的界面与用户进行交互。在Android开发中通常将Activity或者Fragment作为View层。
- Model: 对于Model层也是数据层。它区别于MVC架构中的Model,在这里不仅仅只是数据模型。在MVP架构中Model它负责对数据的存取操作,例如对数据库的读写,网络的数据的请求等。
- Presenter:这一层处理着程序各种逻辑的分发,收到View层UI上的反馈命令、定时命令、系统命令等指令后分发处理逻辑交由Model层做具体的业务操作。
封装应用
在我们初次搭建项目的时候,我们需要一个高扩展的架构,要学会未雨绸缪。在使用MVP架构的时候我们要考虑到如何将View层与Pesenter层进行连接,如何将Model层与Presenter进行连接
首先我们需要一个较好的Base基类
public interface BaseView {
//显示进度框
void showDialog();
//隐藏进度框
void dismissDialog();
}
public abstract class BasePresent<T> {
/**
* 持有UI接口的弱引用
*/
protected WeakReference mViewRef;
/**
* 获取数据方法
*/
public abstract void fetch();
/**
* 绑定的方法
* 在onCreate()中调用
*
* @param view
*/
public void attachView(T view) {
mViewRef = new WeakReference(view);
}
/**
* 解绑
* 在onDestroy方法中调用,防止内存泄漏
*/
public void detach() {
if (mViewRef != null) {
mViewRef.clear();
mViewRef = null;
}
onDestroy();
}
//释放资源处理
public abstract void onDestroy();
}
public interface BaseModel {
}
实际应用
直接上代码
public abstract class BaseActivity<V, T extends BasePresent<V>> extends AppCompatActivity {
/**
* P层引用
*/
protected T mPresent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//创建Presenter层
mPresent = createPresent();
//做绑定
mPresent.attachView((V) this);
//P层数据初始化逻辑
mPresent.fetch();
}
/*
* @params
* @name 子类实现具体的构建过程
* @data 2017/11/20 15:39
* @author :MarkShuai
*/
protected abstract T createPresent();
@Override
protected void onDestroy() {
mPresent.detach();
super.onDestroy();
Log.d(TAG, "onDestroy()");
}
}
直接将View层与Present作为泛型传入其中,将View层与Presenter层进行了绑定,每次将BasePresenter层中的初始化数据方法进行了调用,并且在各个生命周期中,做了防止内存泄漏等。这里我只是抽出了MVP的代码部分进行了讲解,demo中对BaseActivity进行了不错的封装,对Android 兼容性的沉浸式、6.0动态授权,和公共方法的抽取都进行了封装
下面接着来看代码
public interface SerViceUpLoadContract {
interface View extends BaseView<SerViceUpLoadContract.Present> {
void setProgress(Integer progress);
}
abstract class Present<T> extends BasePresent<SerViceUpLoadContract.View> {
//绑定服务
abstract void bindPresentService();
//开始更新APK
abstract void startUpLoadAPK(T t);
}
interface Model extends BaseModel {
//绑定服务
void bindModelService(Context context);
//开始下载
void startUpLoad(Context mContext, ProgressListener listener);
//观察者的监听接口回调
interface ProgressListener {
void onSubscribeProgress(Disposable d);
void onNextProgress(Integer progress);
void onErrorProgress(Throwable throwable);
void onCompleteProgress();
}
}
}
每次在我们进行调用的时候我是喜欢写这样的一个接口,因为这样的话后期方便你的管理,你直接从Contract中就看了大概你这个Activity做了些什么事情。
public class SerViceUpLoadModel implements SerViceUpLoadContract.Model {
private DownloadService.DownloadBinder mDownloadBinder;
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mDownloadBinder = (DownloadService.DownloadBinder) service;
}
@Override
public void onServiceDisconnected(ComponentName name) {
mDownloadBinder = null;
}
};
@Override
public void bindModelService(Context context) {
Intent intent = new Intent(context, DownloadService.class);
context.startService(intent);
context.bindService(intent, mConnection, BIND_AUTO_CREATE);//绑定服务
}
@Override
public void startUpLoad(Context mContext, ProgressListener listener) {
File file = new File(mContext.getExternalFilesDir(Environment.DIRECTORY_DOWNLOADS), "test.apk");
boolean isHave = SDCardHelper.fileIsHave(file.getPath());
if (isHave) {
ToastUtils.showToast("存在了");
} else {
if (mDownloadBinder != null) {
long downloadId = mDownloadBinder.startDownload(ContentManager.APK_URL);
startCheckProgress(downloadId, mContext, listener);
}
}
}
//开始监听进度
private void startCheckProgress(long downloadId, Context mContext, ProgressListener listener) {
Observable
.interval(100, 200, TimeUnit.MILLISECONDS, Schedulers.io())//无限轮询,准备查询进度,在io线程执行
.filter(time -> mDownloadBinder != null)
.map(i -> mDownloadBinder.getProgress(downloadId))//获得下载进度
.takeUntil(progress -> progress >= 100)//返回true就停止了,当进度>=100就是下载完成了
.distinct()//去重复
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new ProgressObserver(mContext, listener));
}
//观察者
private class ProgressObserver implements Observer {
private Context mContext;
private ProgressListener listener;
public ProgressObserver(Context mContext, ProgressListener listener) {
this.mContext = mContext;
this.listener = listener;
}
@Override
public void onSubscribe(Disposable d) {
listener.onSubscribeProgress(d);
}
@Override
public void onNext(Integer progress) {
listener.onNextProgress(progress);
}
@Override
public void onError(Throwable throwable) {
listener.onErrorProgress(throwable);
}
@Override
public void onComplete() {
listener.onCompleteProgress();
}
}
}
这个是Mdoel层的使用
接下来继续看代码
public class SerViceUpLoadPresent extends
SerViceUpLoadContract.Present {
private Context mContext;
private SerViceUpLoadContract.Model mModel;
private Disposable mDisposable;//可以取消观察者
public SerViceUpLoadPresent(Context mContext) {
this.mContext = mContext;
mModel = new SerViceUpLoadModel();
}
@Override
public void fetch() {
}
@Override
void bindPresentService() {
mModel.bindModelService(mContext);
}
@Override
void startUpLoadAPK(SerViceUpLoadContract.View view) {
mModel.startUpLoad(mContext, new SerViceUpLoadContract.Model.ProgressListener() {
@Override
public void onSubscribeProgress(Disposable d) {
mDisposable = d;
}
@Override
public void onNextProgress(Integer progress) {
view.setProgress(progress);
}
@Override
public void onErrorProgress(Throwable throwable) {
throwable.printStackTrace();
ToastUtils.showToast("出错了");
}
@Override
public void onCompleteProgress() {
view.setProgress(100);
ToastUtils.showToast("下载完成");
}
});
}
@Override
public void onDestroy() {
if (mDisposable != null) {
//取消监听
mDisposable.dispose();
}
}
}
我们的Present层没有进行去 new 一个View 层,仅仅是将我们的泛型坐入其中,让其使用
接下来是我们的View层
public class SerViceUpLoadActivity extends BaseActivity> implements SerViceUpLoadContract.View {
@BindView(R.id.bt_upload)
Button mButtonUpLoad;
@BindView(R.id.down_progress)
ProgressBar mDownLoadProgress;
@Override
public View bindView() {
return null;
}
@Override
public int bindLayout() {
return R.layout.activity_ser_vice_up_load;
}
@Override
public void initView(View view) {
//绑定下载服务
mPresent.bindPresentService();
}
@Override
public void initDataAfter() {
}
@Override
public void setListener() {
mButtonUpLoad.setOnClickListener(this);
}
@Override
public void widgetClick(View v) {
switch (v.getId()) {
case R.id.bt_upload:
mPresent.startUpLoadAPK(this);
break;
}
}
@Override
protected SerViceUpLoadPresent createPresent() {
return new SerViceUpLoadPresent<>(this);
}
@Override
public void showDialog() {
LoadingDialog.show(mContext);
}
@Override
public void dismissDialog() {
LoadingDialog.dismiss(mContext);
}
@Override
public void setProgress(Integer progress) {
mDownLoadProgress.setProgress(progress);
}
}
在我们的View层中我们主要是刷新UI来使用的。
这就是我今天带来的我理解封装MVP,有不足的地方欢迎大家指正批评。
源码地址:https://github.com/MarkMingShuai/AndroidMVPObject
喜欢的话请点个赞,给个start