Android MVP 构架初试

目前讨论MVP MVVM 的架构也来越多,这种构架也很适合Android。研究MVP记录如下

源码地址RxMVP分支Tag02

原有的MVC构架

刚开始接触Android的时候会觉得Android的整个代码架构就是一个MVC。

  • M : 业务层和模型层,相当与javabean和我们的业务请求代码
  • V : 视图层,对应Android的layout.xml布局文件
  • C : 控制层,对应于Activity中对于UI 的各种操作

看起来MVC架构很清晰,但是实际的开发中,请求的业务代码往往被丢到了Activity里面,大家都知道layout.xml的布局文件只能提供默认的UI设置,所以开发中视图层的变化也被丢到了Activity里面,再加上Activity本身承担着控制层的责任。所以Activity达成了MVC集合的成就,最终我们的Activity就变得越来越难看,从几百行变成了几千行。维护的成本也越来越高

新的MVP架构

  • M : 还是业务层和模型层
  • V : 视图层的责任由Activity来担当
  • P : 新成员Prensenter 用来代理 C(control) 控制层

MVP与MVC最大的不同,其实是Activity职责的变化,由原来的C (控制层) 变成了 V(视图层),不再管控制层的问题,只管如何去显示。控制层的角色就由我们的新人 Presenter来担当,这种架构就解决了Activity过度耦合控制层和视图层的问题。

一个简单实践

声明了一个接口,带有请求数据业务的方法

RequestBiz

public interface RequestBiz {
    //请求数据业务
    void requestForData(OnRequestListener listener);
}

RequestBizIml

请求的实现类为了模拟网络请求,开启了一个会sleep 10秒的线程,然后装填请求的数据,通过OnRequestListener 接口回调出去,与我们平时开发的方式一致。

public class RequestBizIml implements RequestBiz {

    @Override
    public void requestForData(final OnRequestListener listener) {

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000*10);
                    ArrayList<String> data = new ArrayList<String>();
                    for(int i = 1 ; i< 10 ; i++){
                        data.add("item"+i);
                    }
                    if(null != listener){
                        listener.onSuccess(data);
                    }
                }catch(Exception e){
                    e.printStackTrace();
                }
            }
        }).start();
    }
}

OnRequestListener

数据请求的回掉接口,声明了成功和失败的方法 。

public interface OnRequestListener {

    void onSuccess(List<String> data);
    void onFailed();
}

到此业务层完成,开始MVP的写法.

由于Activity变成了view层不再去控制界面,但是具体的界面的改变api其实还是由Activity来提供的,所以在写MVP之前需要思考,View层需要哪些方法。

  • 显示loading
  • 隐藏loading
  • listview的初始化
  • 弹出Toast消息

MvpView

写MvpView之前接口需要想想界面有哪些交互,根据业务逻辑来确定

public interface MvpView {
     //显示loading progress
     void showLoading();
     //隐藏loading progress
     void hideLoading();
     //ListView的初始化
     void setListItem(List<String> data);
     //Toast 消息
     void showMessage(String messgae);
}

接下来开始写presenter层, 同样在写presenter之前想想控制层需要哪些方法

  • 网络请求
  • 点击事件
public class MvpPresenter {

    private MvpView mvpView;
    RequestBiz requestBiz;
    private Handler mHandler;

    public MvpPresenter(MvpView mvpView) {
        this.mvpView = mvpView;
        requestBiz = new RequestBizIml();
        mHandler = new Handler(Looper.getMainLooper());
    }

    public void onResume(){
        mvpView.showLoading();
        requestBiz.requestForData(new OnRequestListener() {
            @Override
            public void onSuccess(final List<String> data) {
               //由于请求开启了新线程,所以用handler去更新界面
                mHandler.post(new Runnable() {
                    @Override
                    public void run() {
                        mvpView.hideLoading();
                        mvpView.setListItem(data);
                    }
                });

            }

            @Override
            public void onFailed() {
                mvpView.showMessage("请求失败");
            }
        });
    }


    public void onDestroy(){
        mvpView = null;
    }

    public void onItemClick(int position){
        mvpView.showMessage("点击了item" + position);
    }

}

Presenter完成,现在就剩下一件事,Activity中使用Presenter

完整版MVPActivity

public class MVPActivity extends AppCompatActivity implements MvpView, AdapterView.OnItemClickListener {
    @BindView(R.id.mvp_listview)
    ListView mvpListView;

    MvpPresenter mvpPresenter;
    SweetAlertDialog pd;
    Context mContext;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mContext= this;
        ButterKnife.bind(this);
        mvpListView.setOnItemClickListener(this);
        mvpPresenter = new MvpPresenter(this);
    }

    @Override
    protected void onResume() {
        super.onResume();
        mvpPresenter.onResume();
    }

    @Override
    protected void onDestroy() {
        mvpPresenter.onDestroy();
        super.onDestroy();
    }

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
        mvpPresenter.onItemClick(position);
    }

    @Override
    public void showLoading() {
        if (pd == null) {
            pd = new SweetAlertDialog(mContext, SweetAlertDialog.PROGRESS_TYPE);
            pd.getProgressHelper().setBarColor(Color.parseColor("#A5DC86"));
            pd.setTitleText("Loading");
            pd.setCancelable(true);
        }
        pd.show();
    }

    @Override
    public void hideLoading() {
        pd.hide();
    }

    @Override
    public void setListItem(List<String> data) {
        ArrayAdapter adapter = new ArrayAdapter(MVPActivity.this,
                android.R.layout.simple_list_item_1, data);
        mvpListView.setAdapter(adapter);
    }


    @Override
    public void showMessage(String message) {
        Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
    }
}

源码地址:https://github.com/Javen205/RxMVP

转载: http://blog.csdn.net/dantestones/article/details/50899235

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java后端技术栈

Java虚拟机值对象访问以及如何使用对象的引用(2)

对象访问在 Java 语言中无处不在,是最普通的程序行为,但即使是最简单的访问,也会却涉及 Java 栈、 Java 堆、方法区这三个最重要内存区域之间的关联关...

8810
来自专栏Java后端技术

Maven Java EE Configuration Problem 的完美解决办法

  最近在修改项目的时候,发现修改了项目依赖以后会出现如下图:Maven Java EE Configuration Problem 的问题,对于有强迫症的我来...

15520
来自专栏Java后端技术栈

Java虚拟机OOM之虚拟机栈和本地方法栈溢出(4)

(1)如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError 异常; (2)如果虚拟机栈可以动态扩展(当前大部分的 Java ...

11630
来自专栏Java后端技术栈

69 个经典 Spring 面试题和答案

Spring 是个java企业级应用的开源开发框架。Spring主要用来开发Java应用,但是有些扩展是针对构建J2EE平台的web应用。Spring 框架目标...

12030
来自专栏Java后端技术

slf4j介绍以及实现原理窥探

  slf4j(全称是Simple Loging Facade For Java)是一个为Java程序提供日志输出的统一接口,并不是一个具体的日志实现方案,就好...

11120
来自专栏Java后端技术

分模块的maven项目调试时报Source not found的解决办法

  通常在开发中,我们经常会拆分我们的项目为一个个maven子工程,然后用一个父项目进行集成,并且子项目还会继承自父项目。当我们对这些项目进行debug调试的时...

11350
来自专栏Java后端技术栈

QQ登录网站接入功能实现--非官方文档搬运

最近第一次使用QQ登录功能,期间遇到这种问题,在网上找了很多资料,大多都是官方的搬运,并没有真正的干料,可能是个人能力问题,遇到了各种麻烦,折腾了几天,最终弄好...

21140
来自专栏Java后端技术栈

Java分布式开发不得不知的Dubbo技术详细介绍

随着互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行,亟需一个治理系统确保架构有条不紊的演进。

15430
来自专栏Java后端技术栈

Java虚拟机之垃圾收集器(5)

(1)Java 内存运行时区域的各个部分,其中程序计数器、虚拟机栈、本地方法栈三个区域随线程而生,随线程而灭;栈中的栈帧随着方法的进入和退出而有条不紊地执行着出...

8440
来自专栏Java后端技术栈

Java 虚拟机内存区域划分详解(1)

JVM,java virtual machine, 即Java虚拟机,是运行java class文件的程序。

11940

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励