热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

AndroidMVP模式实战教程

这篇文章主要为大家详细介绍了AndroidMVP模式实战教程,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、什么是MVP

在网上找了些资料,整理如下:

  MVP是模型(Model)、视图(View)、主持人(Presenter)的缩写,分别代表项目中3个不同的模块。

  模型(Model):负责处理数据的加载或者存储,比如从网络或本地数据库获取数据等;
  视图(View):负责界面数据的展示,与用户进行交互;
  主持人(Presenter):相当于协调者,是模型与视图之间的桥梁,将模型与视图分离开来。

如下图所示,View与Model并不直接交互,而是使用Presenter作为View与Model之间的桥梁。其中Presenter中同时持有Viwe层以及Model层的Interface的引用,而View层持有Presenter层Interface的引用。当View层某个界面需要展示某些数据的时候,首先会调用Presenter层的某个接口,然后Presenter层会调用Model层请求数据,当Model层数据加载成功之后会调用Presenter层的回调方法通知Presenter层数据加载完毕,最后Presenter层再调用View层的接口将加载后的数据展示给用户。这就是MVP模式的整个核心过程。

在MVP模式里通常包含4个要素:

     (1)View:负责绘制UI元素、与用户进行交互(在Android中体现为Activity);
     (2)View interface:需要View实现的接口,View通过View interface与Presenter进行交互,降低耦合,方便进行单元测试;
     (3)Model:负责存储、检索、操纵数据(有时也实现一个Model interface用来降低耦合);
     (4)Presenter:作为View与Model交互的中间纽带,处理与用户交互的负责逻辑。

二、MVP模式的使用

上面一张简单的MVP模式的UML图,从图中可以看出,使用MVP,至少需要经历以下步骤:

1.创建IPresenter接口,把所有业务逻辑的接口都放在这里,并创建它的实现PresenterCompl(在这里可以方便地查看业务功能,由于接口可以有多种实现所以也方便写单元测试)

2.创建IView接口,把所有视图逻辑的接口都放在这里,其实现类是当前的Activity/Fragment

3.由UML图可以看出,Activity里包含了一个IPresenter,而PresenterCompl里又包含了一个IView并且依赖了Model。Activity里只保留对IPresenter的调用,其它工作全部留到PresenterCompl中实现

4.Model并不是必须有的,但是一定会有View和Presenter

三、一个简单的MVP示例

实现一个简单的用户登录

效果如下:

先来看看java文件的目录结构:

1.Baen包

UserBean

package com.example.mvpe.bean; 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public class UserBean { 
 private String username; 
 private String userpwd; 
 
 public UserBean() { 
 } 
 
 public UserBean(String username, String userpwd) { 
 this.username = username; 
 this.userpwd = userpwd; 
 } 
 
 public String getUsername() { 
 return username; 
 } 
 
 public void setUsername(String username) { 
 this.username = username; 
 } 
 
 public String getUserpwd() { 
 return userpwd; 
 } 
 
 public void setUserpwd(String userpwd) { 
 this.userpwd = userpwd; 
 } 
} 

2.Model包

IUserModel类

package com.example.mvpe.model; 
 
import com.example.mvpe.bean.UserBean; 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public interface IUserModel { 
 
 void setSid(int cid); 
 UserBean load(int sid); 
 void setUsername(String username); 
 void setUserpwd(String userpwd); 
} 

UserModelimp类

package com.example.mvpe.model; 
 
import android.provider.Settings; 
import android.util.Log; 
import android.util.SparseArray; 
 
import com.example.mvpe.bean.UserBean; 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public class UserModelimp implements IUserModel{ 
 
 private static final String ACTIVITY_TAG="UserModelimp"; 
 private String username; 
 private String userpwd; 
 private int sid; 
 private SparseArray su=new SparseArray(); 
 
 @Override 
 public void setSid(int cid) { 
 sid=cid; 
 } 
 
 @Override 
 public UserBean load(int sidy) { 
 sid=sidy; 
 UserBean ub=su.get(sid,new UserBean("not found", 
  "not found")); 
 return ub; 
 } 
 
 
 @Override 
 public void setUsername(String usernamey) { 
 username=usernamey; 
 } 
 
 @Override 
 public void setUserpwd(String userpwdy) { 
 userpwd=userpwdy; 
 UserBean usb=new UserBean(username,userpwd); 
 System.out.print(username+"----"+userpwd+"----"+sid); 
 Log.d(UserModelimp.ACTIVITY_TAG, "Start_OK"+username+"----"+userpwd+"----"+sid); 
 su.append(sid,usb); 
 } 
} 

3.persenter包

UserPersenter类

package com.example.mvpe.presenter; 
 
import android.util.Log; 
import android.view.View; 
 
import com.example.mvpe.bean.UserBean; 
import com.example.mvpe.model.IUserModel; 
import com.example.mvpe.model.UserModelimp; 
import com.example.mvpe.view.IUserView; 
 
 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public class UserPersenter { 
 
 private static final String ACTIVITY_TAG="UserPersenter"; 
 
 private IUserView iUserView; 
 private IUserModel iUserModel; 
 
 public UserPersenter(IUserView view){ 
 iUserView=view; 
 iUserModel=new UserModelimp(); 
 } 
 
 public void saveuser(int sid,String username,String userpwd){ 
 iUserModel.setSid(sid); 
 iUserModel.setUsername(username); 
 iUserModel.setUserpwd(userpwd); 
 } 
 
 public void loaduser(int sid){ 
 UserBean usb=iUserModel.load(sid); 
 Log.d(ACTIVITY_TAG, "loaduser: "+usb.getUsername()); 
 iUserView.setUsername(usb.getUsername()); 
 iUserView.setUserpwd(usb.getUserpwd()); 
 } 
 
} 

4.view包

IUserView类

package com.example.mvpe.view; 
 
import com.example.mvpe.bean.UserBean; 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public interface IUserView { 
 
 int getID(); 
 
 void setUsername(String username); 
 void setUserpwd(String userpwd); 
 
 String getUserpwd(); 
 String getUsername(); 
} 

UserActivity类

package com.example.mvpe.view; 
 
import android.app.Activity; 
import android.os.Bundle; 
import android.view.View; 
import android.widget.Button; 
import android.widget.EditText; 
 
import com.example.mvpe.R; 
import com.example.mvpe.presenter.UserPersenter; 
import android.view.View.OnClickListener; 
import android.widget.Toast; 
 
/** 
 * Created by fendo on 2017/4/29. 
 */ 
 
public class UserActivity extends Activity implements OnClickListener,IUserView{ 
 
 private EditText userid; 
 private EditText username; 
 private EditText userpwd; 
 private Button read; 
 private Button write; 
 private UserPersenter usp; 
 
 @Override 
 protected void onCreate(Bundle savedInstanceState) { 
 super.onCreate(savedInstanceState); 
 usp=new UserPersenter(this); 
 setContentView(R.layout.activity_main); 
 findviewbyid(); 
 read.setOnClickListener(this); 
 write.setOnClickListener(this); 
 } 
 
 
 public void findviewbyid(){ 
 userid= (EditText) findViewById(R.id.userid); 
 username= (EditText) findViewById(R.id.usertext); 
 userpwd= (EditText) findViewById(R.id.userpwd); 
 read= (Button) findViewById(R.id.read); 
 write= (Button) findViewById(R.id.write); 
 } 
 
 @Override 
 public int getID() { 
 return Integer.parseInt(userid.getText().toString()); 
 } 
 
 @Override 
 public void setUsername(String usernamcc) { 
 username.setText(usernamcc); 
 } 
 
 @Override 
 public void setUserpwd(String userpw) { 
 userpwd.setText(userpw); 
 } 
 
 @Override 
 public String getUserpwd() { 
 return userpwd.getText().toString(); 
 } 
 
 @Override 
 public String getUsername() { 
 return username.getText().toString(); 
 } 
 
 
 @Override 
 public void onClick(View view) { 
 switch (view.getId()) { 
  case R.id.read : 
  usp.loaduser(getID()); 
  Toast.makeText(getApplicationContext(), "点击了读"+getID(), Toast.LENGTH_SHORT).show(); 
 
  break; 
  case R.id.write: 
  usp.saveuser(getID(),getUsername(),getUserpwd()); 
  Toast.makeText(getApplicationContext(), "点击了存"+getID()+"--"+getUsername()+"--"+getUserpwd(), Toast.LENGTH_SHORT).show(); 
  break; 
 } 
 } 
} 

可以看到View只负责处理与用户进行交互,并把数据相关的逻辑操作都扔给了Presenter去做。而Presenter调用Model处理完数据之后,再通过IUserView更新View显示的信息。

下载:源码

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍如何使用arm-eabi-gdb调试Android平台上的C/C++程序。通过具体步骤和实用技巧,帮助开发者更高效地进行调试工作。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文介绍如何在 Xcode 中使用快捷键和菜单命令对多行代码进行缩进,包括右缩进和左缩进的具体操作方法。 ... [详细]
  • 本文介绍了一款用于自动化部署 Linux 服务的 Bash 脚本。该脚本不仅涵盖了基本的文件复制和目录创建,还处理了系统服务的配置和启动,确保在多种 Linux 发行版上都能顺利运行。 ... [详细]
  • 在Linux系统中配置并启动ActiveMQ
    本文详细介绍了如何在Linux环境中安装和配置ActiveMQ,包括端口开放及防火墙设置。通过本文,您可以掌握完整的ActiveMQ部署流程,确保其在网络环境中正常运行。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 如何在WPS Office for Mac中调整Word文档的文字排列方向
    本文将详细介绍如何使用最新版WPS Office for Mac调整Word文档中的文字排列方向。通过这些步骤,用户可以轻松更改文本的水平或垂直排列方式,以满足不同的排版需求。 ... [详细]
  • 本文总结了在使用Ionic 5进行Android平台APK打包时遇到的问题,特别是针对QRScanner插件的改造。通过详细分析和提供具体的解决方法,帮助开发者顺利打包并优化应用性能。 ... [详细]
  • 理解存储器的层次结构有助于程序员优化程序性能,通过合理安排数据在不同层级的存储位置,提升CPU的数据访问速度。本文详细探讨了静态随机访问存储器(SRAM)和动态随机访问存储器(DRAM)的工作原理及其应用场景,并介绍了存储器模块中的数据存取过程及局部性原理。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 几何画板展示电场线与等势面的交互关系
    几何画板是一款功能强大的物理教学软件,具备丰富的绘图和度量工具。它不仅能够模拟物理实验过程,还能通过定量分析揭示物理现象背后的规律,尤其适用于难以在实际实验中展示的内容。本文将介绍如何使用几何画板演示电场线与等势面之间的关系。 ... [详细]
  • 本文介绍如何通过Windows批处理脚本定期检查并重启Java应用程序,确保其持续稳定运行。脚本每30分钟检查一次,并在需要时重启Java程序。同时,它会将任务结果发送到Redis。 ... [详细]
  • MySQL中枚举类型的所有可能值获取方法
    本文介绍了一种在MySQL数据库中查询枚举(ENUM)类型字段所有可能取值的方法,帮助开发者更好地理解和利用这一数据类型。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
author-avatar
王琰杰昔凛
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有