一文形象生动地让你理解什么是MVP架构模式,看不懂你砍我

2023-10-19 20:20

本文主要是介绍一文形象生动地让你理解什么是MVP架构模式,看不懂你砍我,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

本文将会以讲故事的形式加深理解。

在Android传统模式开发中,随着Activity的功能增多,其代码也在不断增加。那么我们很可能会发现,如果我们再要往Activity里边加一个功能,而恰好这个功能又要读数据库又要请求Http,
爆炸!加这么个功能要1000行代码(夸张),那么我们的Activity的代码就会变得十分臃肿,你没法快速地完成一个功能代码的修改。
因此,MVP架构模式正是解决这样一种情况的好方法

什么是MVP架构模式

在很多博客当中是这么解释MVP的,

  • M:Model,实体层,用于实现具体的复杂的逻辑
  • V:View,视图层,用于展示数据、交互等
  • P:Presenter,逻辑控制层,用于持有Model和View的实例

然后就开始上一堆代码了。。。头疼


因此我写下这篇文章打算用一种形象生动的方法

首先你需要接受这么一个设定

  • View:你
  • Presenter:王大妈
  • Model:工具人

这么设定只是为了帮助理解,不要问为什么是王大妈而不是张大妈,这只是一个虚的东西。


正式开始

故事:
想要去买一瓶阔乐,以前的时候都是你自己一个人去买的,但是现在有一个王大妈出现了。
她说:“你把钱给我,可乐我帮你买,不多收费!”
你听了之后,这么划算的事情,那就这么定了。
于是你把钱给了王大妈,也把手机号码给了她。

但是我们都知道,天下没有免费的午餐,对于王大妈来说,这是个赚钱的机会。
王大妈知道一个低价买阔乐的渠道->工具人,他可以自己生产阔乐,因此价格便宜,她可以赚差价。
于是,王大妈找到了工具人,他们各自给了对方联系方式。

从此以后,只要想喝阔乐,你就拨打王大妈的手机号跟她说。
而王大妈知道之后,她又打电话给工具人,向他要阔乐。
工具人听到之后,开始卖命地生产阔乐。

一段时间之后,阔乐搞出来了,工具人打电话通知王大妈,王大妈知道后又打电话通知

如此一来,阔乐就到手了。你也完全不知道阔乐是怎么来的,但确实到你手里了。


引入术语

这样以故事的形式来解释,我个人感觉会比较好理解。现在我们在故事的基础上将术语引入。

  • 你:View
  • 王大妈:Presenter
  • 工具人:Model

从这种架构模式来看,View想要拿到数据显示,这个流程的逻辑将会变得十分清晰!

Presenter作为一个中间人,需要分两头联系View和Model。

而Model,只管把数据搞出来,再通知Presenter就好了,Presenter会做剩下的工作。


用代码举例

从上文可得知,王大妈工具人都是十分具体的,但是如果另外一个张大爷也可以帮你完成代购,而且速度很快呢?

相信你一定也有了启发,这个时候 只要能代购到可乐,哪个中间人都行(只是效率问题)。

那有一天你不想喝阔乐了,想喝芬达了呢?

进一步处理

因此我们将这个Presenter(中间人)抽象,抽象成一个接口,只要能帮我们买到可乐,就能做这个中间人
在中间人接口中,我们需要指定,这个中间人需要帮我们做些什么。(代购些啥)

同理,我们将Model(中间人)也抽象成一个接口,只要能干又苦又累的活都能做工具人
在工具人接口中,我们需要指定,这个工具人要帮我们搞些什么东西。(生产些啥)

呢?(View)也可以抽象成一个接口,我们需要指定,这个接口要些啥。(需要啥)

所以View需要啥→Presenter代购啥→Model生产啥,随后,
Model生产完成→Presenter代购完成→View心愿完成


这个流程都是十分清晰的,接下来直接上代码了

上代码,新建3个接口

新建3个包,分别为model,presenter和view
在这里插入图片描述


在presenter包中,添加MiddlePeople接口(中间人接口)

package cn.daccc.mvpdemo.presenter;/*** 中间人接口,只要能帮我们买到可乐,能通知我们就行*/
public interface MiddlePeople{//买可乐void buyCola();//购买成功后调用该方法void buySucceed();
}

在model包中,添加ToolPeople接口(工具人接口)
package cn.daccc.mvpdemo.model;/*** 工具人接口,只要能生产可乐就行*/
public interface ToolPeople{//生产可乐void produceCola();}

在view包中,添加Me这个接口,实现喝可乐的心愿(中间人买回来之后我们就可以喝了)
package cn.daccc.mvpdemo.view;/*** 实际上不一定是你想喝,你同学也想喝呢?就假装他也是“你”*/
public interface Me {//喝可乐void drinkCola();
}

我们添加了上面的3个接口,都是没有具体的实现的,也就是说,也就嘴上说说而已,没有实际行动
因此我们来建立真真实实的类,去完成上述行为

上代码,新建2个类

我们指定HanHan(憨憨)作为工具人,让他去帮我们生产可乐。

在model包中新建一个HanHan类

这里注意了,这个憨憨要符合工具人的特性,因此他要实现ToolPeople接口的所有方法,做工具人做的事。
因此,他要实现接口中的produceCola()方法

package cn.daccc.mvpdemo.model;
import android.util.Log;
/*** 工具人憨憨,他要符合工具人的特性才能帮我们做事,因此要实现Tool接口*/
public class HanHan implements ToolPeople {@Overridepublic void produceCola() {Log.d("ToolPeople","生产可乐好累啊!");}
}

你会发现,HanHan生产完了,怎么通知呢?所以,HanHan要知道中间人的联系方式,因此HanHan要在中间人联系他的时候保存好中间人的联系方式,然鹅HanHan并不知道中间人具体是谁,所以要用接口类型来保存中间人。

package cn.daccc.mvpdemo.model;
import android.util.Log;
import cn.daccc.mvpdemo.presenter.MiddlePeople;
/*** 工具人憨憨,他要符合工具人的特性才能帮我们做事,因此要实现Tool接口*/
public class HanHan implements ToolPeople {//保存中间人的联系方式private MiddlePeople middlePeople;public HanHan(MiddlePeople middlePeople){this.middlePeople = middlePeople;}@Overridepublic void produceCola() {Log.d("ToolPeople","生产可乐好累啊!");//HanHan生产好了,告诉中间人TA购买成功了middlePeople.buySucceed();}
}

在presenter包中新建WangDaMa类

现在已经有了一个具体的工具人了,但是我们还差个具体的中间人
因此我们指定WangDaMa(王大妈)为中间人,这王大妈既要有“我”的联系方式,也要找到一个符合要求的工具人并且保存工具人的联系方式,因此她找到了HanHan

package cn.daccc.mvpdemo.presenter;import android.util.Log;import cn.daccc.mvpdemo.model.ToolPeople;
import cn.daccc.mvpdemo.view.Me;/*** 中间人,指定王大妈是中间人,因此她要做所有中间人该做的事情*/
public class WangDaMa implements MiddlePeople {private Me me;private ToolPeople toolPeople;//保存“我”的联系方式,并指派HanHan为工具人,保存他的联系方式public WangDaMa(Me me){this.me=me;toolPeople = new HanHan(this);}@Overridepublic void buyCola() {Log.d("MiddlePeople","我来帮你买可乐");//让工具人去生产可乐toolPeople.produceCola();}@Overridepublic void buySucceed() {//买成了,通知“我”可以开始嚯阔乐啦me.drinkCola();}
}

在MainActivity中实现Me这个接口
我们找一个中间人做代购,因此我找到了王大妈。
我只要想喝可乐只需要点击一下按钮,就让中间人帮我买可乐,买完之后再告诉我就可以了

public class MainActivity extends AppCompatActivity implements Me{private MiddlePeople middlePeople = new WangDaMa(this);@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Button btn_drink = findViewById(R.id.btn_drink);btn_drink.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {//点击按钮,就让中间人帮我们买可乐middlePeople.buyCola();}});}/*** 实现Me接口中的drinkCola方法,让喝可乐这件事情有一个更加具体的描述*/@Overridepublic void drinkCola() {Toast.makeText(this, "这阔乐真他*的好嚯!", Toast.LENGTH_SHORT).show();}
}

结果展示

在这里插入图片描述

诶,就很舒服

弊端

打住,舒服是暂时的,到底有没有必要为了喝个可乐去搞那么多乱七八糟的什么中间人工具人呢?

答案就是:请使用长远的眼光,来决定在具体Android开发项目中到底需不需要使用MVP架构模式

文章中这样的设计模式会增加很多的类的接口,这是一个弊端,当然可以优化!
由于篇幅原因,我会把优化后的MVP架构个人理解放到下一篇博文中,敬请期待。

总结

从上边的故事和例子来看,是不是都理解了呢?如果还不理解,你来砍我(顺着网线来)。
哈哈,开个玩笑,我们来总结一下

  • ToolPeople:实际上是Model,它做很多又苦又累的工作,因此把业务代码放在这里
  • MiddlePeopl:实际上是Presenter,它做一个中间人的角色,可以执行少量的业务代码,比如简单判断这件事情能不能干的成
  • Me:实际上是View,喝可乐这件事情最终体现在View中,但是它无需知道可乐是怎么来的,它只需要委派中间人去实现就可以了
最终总结

这里的什么憨憨,王大妈,“我”。都是为了帮助理解而已,相信认真看完文章的人,应该都理解了,因此这些类的名称,这些接口的名称,都应该按照规范来命名

编程很少是一个人的工作,所以学习如何架构代码是进阶的必经之路。本文分享了我对MVP架构的一些浅见,希望能够帮助大家理解。

如果文章有误,恳请指出,十分感谢!

机会,总是留给有准备的人。

共勉

这篇关于一文形象生动地让你理解什么是MVP架构模式,看不懂你砍我的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/242137

相关文章

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

一文详解Git中分支本地和远程删除的方法

《一文详解Git中分支本地和远程删除的方法》在使用Git进行版本控制的过程中,我们会创建多个分支来进行不同功能的开发,这就容易涉及到如何正确地删除本地分支和远程分支,下面我们就来看看相关的实现方法吧... 目录技术背景实现步骤删除本地分支删除远程www.chinasem.cn分支同步删除信息到其他机器示例步骤

mysql中的服务器架构详解

《mysql中的服务器架构详解》:本文主要介绍mysql中的服务器架构,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、mysql服务器架构解释3、总结1、背景简单理解一下mysqphpl的服务器架构。2、mysjsql服务器架构解释mysql的架

一文详解Java Stream的sorted自定义排序

《一文详解JavaStream的sorted自定义排序》Javastream中的sorted方法是用于对流中的元素进行排序的方法,它可以接受一个comparator参数,用于指定排序规则,sorte... 目录一、sorted 操作的基础原理二、自定义排序的实现方式1. Comparator 接口的 Lam

k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)

《k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)》本文记录在K8s上运行的MySQL/MariaDB备份方案,通过工具容器执行mysqldump,结合定时任务实... 目录前言一、获取需要备份的数据库的信息二、备份步骤1.准备工作(X86)1.准备工作(arm)2.手