MVP架构简介

MVP模式是由著名的MVC模式演变过来的,Android应用开发本身就是MVC模式,其中xml的布局是View层,Activity就相当于Controller。然而实际开发过程中,Activity经常不能纯粹的充当Controller,因为既要负责渲染view,又要处理业务逻辑,所以经常看到一个Activity有上千行代码,代码臃肿难懂,维护起来也非常吃力。这时候就需要简化Activity的职责,让它只负责界面的绘制和渲染,再抽出一层来负责处理业务逻辑,这就演变出了MVP模式。


MVP模式分为三层:Model层, View层,Presenter层。

View层:也就是视图层,负责展示数据,以及用户与界面的交互,提供良好的用户体验,View层一般是Activity或者Fragment。

Model层: 数据处理层,为用户界面提供基本的数据支持,例如数据库的增删改查,网络操作。

Presenter层:它是连接view 层和model层的中间层,处理业务逻辑。它从Model层获取数据,处理之后交给view层显示。

关于MVP模式,Google官方也给出了一些架构的实现。源码地址:https://github.com/googlesamples/android-architecture. 


实例分析

我自己写了一个简单的Demo用来加深印象,此demo是模拟从服务器load天气数据,然后在对数据进行处理,最终显示在界面上,主要是用来说明三个层之间的关系,并没有实现具体的代码。目录结构如下:


ios mvp架构实践 mvp架构模式_数据

首先Model层,对应的是WeatherDataSource,一个单例的类,是对网络请求的封装,代码如下:

public class WeatherDataSource {

    private static final int MSG_LOAD_CHINA_DATA = 1;
    private static final int MSG_LOAD_GLOBAL_DATA = 2;
    private static WeatherDataSource sWeatherDataSource;
    private Context mContext;

    private MyHandler mHandler = null;

    class MyHandler extends Handler {

        public MyHandler(Looper looper) {
            super(looper);
        }

        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            switch (msg.what) {
                case MSG_LOAD_CHINA_DATA:
                    doLoadChinaData((ILoadWeatherDataListener)msg.obj);
                    break;
                case MSG_LOAD_GLOBAL_DATA:
                    break;
            }
        }
    }

    private WeatherDataSource(Context context) {
        mContext = context;
        //创建一个工作线程,用于从网络load数据
        HandlerThread handlerThead = new HandlerThread("work thread");
        mHandler = new MyHandler(handlerThead.getLooper());
        handlerThead.start();
    }

    public static synchronized WeatherDataSource getInstance(Context context) {
        if(sWeatherDataSource == null) {
            sWeatherDataSource = new WeatherDataSource(context);
        }
        return sWeatherDataSource;
    }

    /**
     * 从中央气象局接口load数据
     * @param listener
     */
    public void loadChinaData(ILoadWeatherDataListener listener){

        if(mHandler != null) {
            mHandler.sendMessage(mHandler.obtainMessage(MSG_LOAD_CHINA_DATA, listener));
        }
    }

    private void doLoadChinaData(ILoadWeatherDataListener listener) {
        //load data...
    }

    public interface ILoadWeatherDataListener {
        void onLoadSuccess(String data);
        void onLoadFailed();
    }

}

其中,在构造方法中,创建了一个HandlerThread,用于异步请求网络数据,请求回来之后,通过ILoadWeatherDataListener回调做相应操作。

Presenter层与View层提供的接口,定义了View层,以及Presenter层对外提供哪些接口,代码如下,

public interface LoadWeatherData {

    public interface BaseView {
        public void showData(String data);
    }

    public interface BasePresenter {
        public String loadChinaData();
    }
}

其中,View层提供显示数据的接口,而Presenter提供查询数据的接口。

Presenter层,对应的是LoadDataPresenter类,代码如下,

public class LoadDataPresenter implements WeatherDataSource.ILoadWeatherDataListener{

    private Context mContext;
    private LoadWeatherData.BaseView mBaseView;

    @Override
    public void onLoadFailed() {
        mBaseView.showData(null);
    }

    @Override
    public void onLoadSuccess(String data) {
        mBaseView.showData(data);
    }

    public LoadDataPresenter(Context context, LoadWeatherData.BaseView baseView) {
        mContext = context;
        mBaseView = baseView;
    }

    public void loadChinaWeatherData() {
        WeatherDataSource.getInstance(mContext).loadChinaData(this);
    }
}

此类实现了ILoadWeatherDataListener接口,并且依赖view层,当load成功或者失败之后,调用View层的showData方法。


View层,对应的就是MainActivity,代码如下:

public class MainActivity extends Activity implements LoadWeatherData.BaseView {

    private static final String TAG = "MainActivity";

    private LoadDataPresenter mPresenter;

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

        mPresenter = new LoadDataPresenter(this,this);

        mPresenter.loadChinaWeatherData();

    }

    @Override
    public void showData(String data) {
        if(data != null) {
            Log.d(TAG, "----showData-----" + data);
        } else {
            Log.d(TAG, "----showData-----data is null" );
        }

    }
}

在onCreate方法里,初始化LoadDataPresenter,并且调用loadChinaWeatherData方法,而且实现了showData方法,来显示数据。一般情况下View层都会与用户交互,下拉刷新之类的操作,不同的操作调用Presenter相应的接口。

以上就是整个Demo的简单说明。

总之,使用MVP架构可以使代码层级之间的结构更加单一清晰,从而让我们的项目更好维护和扩展。