MMP???不是,是MVP

本篇文章代码引用GitHub上zhihu项目,https://github.com/yiyibb/Zhihu。在这里跟初学者强烈推荐一下,这个项目功能虽然简单,只是一个简单阅读功能,但是我很佩服开发者的代码封装能力,做的真得很完美。采用的是MVP+retrofit+rxjava架构。今天主要看他的架构分析。

MVP

我们在初学android的时候,用的架构都是MVC,何为MVC呢。model,view,control。简单来讲model就是数据层,view就是activity,control就是逻辑处理。
开始学的MVC很容易理解,所有层都能在activity上看到,逻辑也很容易看清。activity能够和model,view两者互相接触。但同时也带来相当的困扰,耦合性太大。你想啊,只要activity改动很容易牵连到方面,而且activity任务量也比较大。
由此从MVC衍生出MVP,MVP很大程度上解决了耦合性的问题,view层不再能直接接触model,我感觉这个应该是最大的亮点。当然还有其他优势,比如更方便进行单元测试等等。具体的网上查一查就明白了。

retrofit+rxjava

一个是网络请求,一个是异步处理,由于本文主要讲架构分析,这些用法不会赘述。推荐读者看一看开发者抛物线的文章。上面说过了我是引用别人的代码,其实刚开始我用的就是MVC开发模式,没有全部按照他的写。很蹩脚,重构代码真的很费劲。当初也只是想切身体验一下两个架构的区别,结果越整越乱,重点是我并不是很熟悉MVP的结构模型

架构搭建

首先分析一下功能,从api拿到json数据,然后用retrofit开始解析,拿到数据后用rxjava更新UI集合。来看看整体架构,代码是从项目里摘出来的,功能也已经实现,链接如下https://github.com/ZhangPeng0220/MVP_Retrofit_Rxjava

image.png

整体有一个mvp架构放在了 mvp_base文件夹里面,drawerMvp是整体项目的一个侧滑栏模块, DrawerMainContract类 里包含了drawer模块需要的model,view,presenter。api是model数据连接,common里也是一些基类。接下来可以分层分析。

Drawer MVP模块

这里主要介绍DrawerMainContract接口类

public interface DrawerMainContract {

    interface Model extends BaseModel {
        Observable getOtherThemeList();
    }

    interface View extends BaseView {
        void loadOtherThemeList(ThemesEntity themesEntity);
    }

    abstract class Presenter extends BasePresenter {
        abstract void getOtherThemes();
    }
}

这里所有最终的实体类,都会实现DrawerMainContract中的接口。比如mainactivity会实现view接口

View

这里的view层显而易见就是mainActivity

public class MainActivity extends BaseFrameActivity implements BaseView, DrawerMainContract.View{

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

    @Override
    public void loadOtherThemeList(ThemesEntity themesEntity) {
        //处理返回结果
        themesEntity.getLimit();
    }
}

BaseFrameActivity继承自BaseActivity,除了拥有activity的一般生命周期方法以外,它还主要实现了model和presenter的实例化。

public abstract class BaseFrameActivity

extends BaseActivity implements BaseView{ public P mPresenter; public M mModel; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mPresenter = TUtil.getT(this, 0); mModel = TUtil.getT(this, 1); if (this instanceof BaseView) { mPresenter.attachVM(this, mModel); } } @Override protected void onDestroy() { /// if (mPresenter != null) mPresenter.detachVM(); super.onDestroy(); } }

实例化是通过工具类 TUtil实现的,具体代码请读者移步上文的GitHub地址。这里需要注意的有三个点:
第一个是泛型限制,传入的presenter和model都有父类限制。
第二个需要注意的是,在view层只会接触到presenter,不会接触model,为啥这边还要实例化model。这里的model实例其实为presenter准备的。mPresenter.attachVM(this, mModel)这句话是关键。
第三个是baseview这里是空的,这里可以放一些基本的方法,比如initview,initdata

Presenter

public abstract class BasePresenter  {
    public M mModel;
    public V mView;
  
    public void attachVM(V v, M m) {
        this.mModel = m;
        this.mView = v;

    }

    public void detachVM() {
        //mRxManager.clear();
        mView = null;
        mModel = null;
    }
}

BasePresenter主要用于刚刚view和model的实例化。由于这个类有实体方法并未使用接口。接下来就是DrawerMainContract.Presenter继承了这个抽象方法,具体实现类是DrawerMainPresenter

public class DrawerMainPresenter extends DrawerMainContract.Presenter{

    @Override
     public void getOtherThemes() {
        mModel.getOtherThemeList()
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber() {
                    @Override
                    public void onCompleted() {

                    }

                    @Override
                    public void onError(Throwable e) {
                       // mView.onRequestError("数据加载失败ヽ(≧Д≦)ノ");
                    }

                    @Override
                    public void onNext(ThemesEntity themesEntity) {
                        mView.loadOtherThemeList(themesEntity);
                    }
                });
    }
}

mModel.getOtherThemeList()这个方法是属于model层的,会返回一个Observable对象,这就属于RxJava的内容了,当得到了具体数据,会在onNext方法里面调用view更新数据的方法,此时mView已经在BaseFrameActivity中的mPresenter.attachVM方法实例化了。

Model

这一层主要是用于连接数据,调用数据。

public class DrawerMainModel implements DrawerMainContract.Model{
    @Override
    public Observable getOtherThemeList() {
        return Networks.getInstance().getThemeApi().getThemes();
    }
}

DrawerMainModel实现的接口没啥可讲的,方法可以看一看

public MyService getThemeApi() {
        return mThemeApi == null ? configRetrofit(MyService.class) : mThemeApi;
    }

    private  T configRetrofit(Class service) {
        retrofit = new Retrofit.Builder()
                .baseUrl("http://news-at.zhihu.com")
                .client(configClient())
                .addConverterFactory(GsonConverterFactory.create())
                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                .build();
        return retrofit.create(service);
    }

    private OkHttpClient configClient() {
        OkHttpClient.Builder okHttpClient = new OkHttpClient.Builder()
                .connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS);
        return okHttpClient.build();
    }

这是network里面的实现方法,getThemeApi()方法主要是拿到myservice的实例,继而调用myservice中的getThemes(),拿到json数据。

总结

整体流程就是上面得了,其实不算难,但是作者的封装风格我比较喜欢。流程图如下。可以看出 和MVC相比较是清晰了很多。


image.png

感兴趣的同学可以关注一下我的微信公众号,或者微信搜索 开发 Android的小学生


公众号.jpg

你可能感兴趣的:(Android)