手把手教你搭建简单的mvp框架

mvp java retrofit gson okhttp


对MVP的文字描述介绍

MVC(Model - View - Controller)

  • Model : 数据(JavaBean实体类,用于保存实例数据)
  • View : UI界面,和用户交互(向用户展示数据以及接收用户的输入)
  • Controller : 更新UI界面和数据实例

例如:View层接受用户的输入,然后通过Controller修改对应的Model实例;同时,当Model实例的数据发生变化的时候,需要修改UI界面,可以通过Controller更新界面。(View层也可以直接更新Model实例的数据,而不用每次都通过Controller,这样对于一些简单的数据更新工作会变得方便许多。)

MVP

MVP.png

MVP把Activity中的UI逻辑抽象成View接口,把业务逻辑抽象成Presenter接口,Model类还是原来的Model。
在MVP模式中Activity的功能就是响应生命周期和显示界面,具体其他的工作都丢到了Presenter层中进行完成,Presenter其实是Model层和View层的桥梁。

  • M(model) :数据层,用来放数据的处理(比如网络请求,缓存等)
  • V(view) : 负责UI具体实现展现。比如Presenter派发过来一个动作是showDialog显示进度命令,那么我们这个View就负责实现具体UI
  • P(presenter) : 负责处理业务逻辑代码,处理Model数据,然后将处理完的数据分发到View层
MVP.png

关于android的MVP模式其实一直没有一个统一的实现方式,不同的人由于个人理解的不同,进而产生了很多不同的实现方式,其实很难去说哪一种更好,哪一种不好,针对不同的场合,不同的实现方式都有各自的优缺点。
在这里,我介绍的MVP是Google提出的一种MVP实现方式。

APP——Dreamer

这个app要实现的是类似于“周公解梦”的查询功能,点击卡片可以看详情

Dreamer.png

手撸代码!!!

准备工作
  • 在 build.gradle(Module: app )里添加依赖并 sync
  • 下载GsonFormat 插件,便于后续根据拿到的 json 数据直接生成使用该插件生成类
  • api 接口(此api接口仅供本人自己使用哦~)
  • android manifests 中添加网络权限

ps 下面以此app的search为例子讲 MVP 框架

1.Contract

加入了契约类来统一管理view与presenter的所有的接口,这种方式使得view与presenter中有哪些功能,一目了然,维护起来也方便,实例如下:

package com.example.chenshuyu.dreamer.search;

import...

public interface DreamerSearchContract {
    interface DreamerSearchUIView{
        //当网络错误等原因获取搜索结果没有成功时
        void onError();
        //成功获得搜索结果,且结果不为空时
        void updateRV(List<Dreamer.DataBean> dataBeans);
        //成功获得搜索结果,但是并没有搜出来东西时
        void onNull();
    }

    interface DreamerSearchPresenter{
        //根据搜索输入的获keyword获得搜索结果的
        void getSearchDream(String keyword);
    }
}
2.Model
package com.example.chenshuyu.dreamer.search;

import ...

public class DreamSearchModel {

    private Retrofit retrofit = new Retrofit.Builder()
            //设置数据解析器
            .addConverterFactory(GsonConverterFactory.create())
            //设置网络请求的Url地址
            .baseUrl("https://api.shenjian.io/")
            .build();

    // 创建网络请求接口的实例
    private RetrofitService api = retrofit.create(RetrofitService.class);

    //通过model层的update方法获得网络请求拿到的数据
    public Call<Dreamer> update(String keyword){
        return api.getSearch("cfcde6656c3d1b67ecbecf400592d05e",keyword);
    }
}
3.Presenter
package com.example.chenshuyu.dreamer.search;

/**
 * presenter接口的具体实现类
 */
public class DreamSearchPresenterImpl implements DreamerSearchContract.DreamerSearchPresenter {
    
    /**
     * presenter应该持有 view 层和 model 层的引用
     * 这样才能完成两层之间的逻辑交互
     * 同时使 view 层和 model 层完全隔离开来
     */
    private DreamerSearchContract.DreamerSearchUIView searchUIView;
    private DreamSearchModel dreamSearchModel = new DreamSearchModel();

    /**
     * presenter 层对应的类持有的 view 层对应的类是没有办法在 presenter 内部实例化的(此时的view是有方法但是没有具体实现的接口)
     * view 层的具体实现是 activity 继承 view 层,并重写 view 层的所有方法,即 activity 就是 view 层
     * 故成员对象 view 的实例化对象是在activity中传给presenter的
     * 所以 presenter 的构造函数中应该传入 view
     * model 层是有具体实现类的,并且已经在 presenter 类的内部实例化了,这样才能拿到 model 的具体数据,进行操作
     * @param searchUIView
     */
    public DreamSearchPresenterImpl(DreamerSearchContract.DreamerSearchUIView searchUIView){
        this.searchUIView = searchUIView;
    }

    /**
     * view 层和 model 层的逻辑交互,根据model的数据,执行相关的view层操作
     * @param keyword
     */
    @Override
    public void getSearchDream(String keyword) {
        Call<Dreamer> call = dreamSearchModel.update(keyword);
        call.enqueue(new Callback<Dreamer>() {
            
            //发送网络请求成功
            @Override
            public void onResponse(Call<Dreamer> call, Response<Dreamer> response) {
                Dreamer dreamer = response.body();
                if (dreamer.getData() == null){
                    searchUIView.onNull();
                }else {
                    searchUIView.updateRV(dreamer.getData());
                }
            }

            //没有成功
            @Override
            public void onFailure(Call<Dreamer> call, Throwable t) {
                searchUIView.onError();
                t.printStackTrace();
            }
        });
    }
}
4.View
package com.example.chenshuyu.dreamer.search;

import...

public class SearchActivity extends AppCompatActivity implements DreamerSearchContract.DreamerSearchUIView{
    private String keyword;
    private EditText editText;
    private ImageView imageView;
    private RecyclerView recyclerView;
    private SearchAdapter searchAdapter;
    private LinearLayout linearLayout;
    private List<Dreamer.DataBean> dataBeans = new ArrayList<>();
    private DreamerSearchContract.DreamerSearchPresenter presenter = new DreamSearchPresenterImpl(this);

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_search);
        initId();
        initRV();

        linearLayout.setVisibility(View.GONE);
        editText.setHint("请输入你想搜索的关键字");
        // 点击搜索,从EditTex获得搜索keyword,并通过presneter获得搜索结果
        imageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                keyword = editText.getText().toString();
                presenter.getSearchDream(keyword);
            }
        });
    }

    // 绑定控件
    private void initId(){
        editText = findViewById(R.id.search_edit);
        imageView = findViewById(R.id.search_img);
        recyclerView = findViewById(R.id.search_rv);
        linearLayout = findViewById(R.id.search_no);
    }

    // 初始化recyclerview
    private void initRV(){
        LinearLayoutManager linearLayout = new LinearLayoutManager(this);
        linearLayout.setOrientation(LinearLayoutManager.VERTICAL);
        searchAdapter = new SearchAdapter(dataBeans,this);
        recyclerView.setLayoutManager(linearLayout);
        recyclerView.setAdapter(searchAdapter);
    }

    // 重写view的方法
    @SuppressLint("CheckResult")
    @Override
    public void onError() {
        linearLayout.setVisibility(View.GONE);
        Toasty.error(this,"你的网络崩溃了,5555~",Toast.LENGTH_LONG).show();
    }

    @Override
    public void updateRV(List<Dreamer.DataBean> dataBeanArrayList) {
        linearLayout.setVisibility(View.GONE);
        searchAdapter.update(dataBeanArrayList);
    }

    @Override
    public void onNull() {
        dataBeans.clear();
        searchAdapter.update(dataBeans);
        linearLayout.setVisibility(View.VISIBLE);
    }
}

代码架构

image.png
  • 每一个activity就是一个mvp框架搭建起来的
  • entity用来存放数据请求获得的类
  • service 统一管理网络请求的接口

想看完整代码的大兄弟和小仙女,可以从github上clone,在此附上我的仓库地址~:
https://github.com/chenshuyuhhh/Dreamer.git

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容