MVP模式 详解 案例
介绍
MVC:
- View:对应于布局文件
- Model:业务逻辑和实体模型
- Controllor:对应于Activity
实际上关于该布局文件中的数据绑定的操作,事件处理的代码都在Activity中,造成了Activity既像View又像Controller程序员对于MVP的普遍的认识是:代码很清晰,不过增加了很多类当将架构改为MVP以后,Presenter的出现,将Actvity视为View层,Presenter负责完成View层与Model层的交互。现在是这样的:
- View 对应于Activity,负责View的绘制以及与用户交互
- Model 依然是业务逻辑和实体模型
- Presenter 负责完成View和Model间的交互
从并不标准的MVC到MVP的一个转变,减少了Activity的职责,简化了Activity中的代码,将复杂的逻辑代码提取到了Presenter中进行处理。与之对应的好处就是,耦合度更低,更方便的进行测试。MVC与MVP的一个区别示意图:其实最明显的区别就是,MVC中是允许Model和View进行交互的,而MVP中很明显,Model与View之间的交互由Presenter完成。还有一点就是Presenter与View之间的交互是通过接口的。
案例
Activity包(view包)
/*** 梳理此Activity所有需要的方法,将这些方法全部封装到一个接口中* @author 白乾涛*/public interface IUserLoginView {public String getUserName();public String getPassword();public void clearUserName();public void clearPassword();public void toMainActivity(User user);public void showFailedError();public void showLoading();public void hideLoading();}/*** 简单来说,对Activity的所有操作(方法)都被剖离在了两个地方,一个是在IUserLoginView中,这里面都是对Activity的一些最基本的操作* 另一个在Presenter中,比较复杂的【业务逻辑】都会放在这里(目的当然是减少Activity的代码逻辑)* 一定要明白,之所以将Activity中的这些方法剖离在IUserLoginView中,是为了在Presenter中能通过操作IUserLoginView来操作Activity(扩展性)** @author 白乾涛*/public class UserLoginActivity extends Activity implements IUserLoginView, OnClickListener {private EditText id_et_username;private EditText id_et_password;private Button id_btn_login;private Button id_btn_clear;private UserLoginPresenter mUserLoginPrestener = new UserLoginPresenter(this);//这里是将IUserLoginView接口的实例(即Activity)传给了Presenterprivate ProgressBar id_progressbar;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_user_login);id_et_username = (EditText) findViewById(R.id.id_et_username);id_et_password = (EditText) findViewById(R.id.id_et_password);id_btn_login = (Button) findViewById(R.id.id_btn_login);id_btn_clear = (Button) findViewById(R.id.id_btn_clear);id_progressbar = (ProgressBar) findViewById(R.id.id_progressbar);id_btn_login.setOnClickListener(this);id_btn_clear.setOnClickListener(this);}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.id_btn_login://这里面的两个方法在Presenter中,当我们需要修改业务逻辑时,只需修改Prestener类中的相应方法即可!mUserLoginPrestener.login();break;case R.id.id_btn_clear:mUserLoginPrestener.clear();break;}}//******************************************************************************************@Overridepublic String getUserName() {return id_et_username.getText().toString();}@Overridepublic String getPassword() {return id_et_password.getText().toString();}@Overridepublic void clearUserName() {id_et_username.setText("");}@Overridepublic void clearPassword() {id_et_password.setText("");}@Overridepublic void toMainActivity(User user) {Toast.makeText(getApplicationContext(), user.getUsername() + "--登录成功", Toast.LENGTH_SHORT).show();}@Overridepublic void showFailedError() {Toast.makeText(getApplicationContext(), "登录失败", Toast.LENGTH_SHORT).show();}@Overridepublic void showLoading() {id_progressbar.setVisibility(View.VISIBLE);}@Overridepublic void hideLoading() {id_progressbar.setVisibility(View.INVISIBLE);}}
presenter包
/*** 这就是传说中的Prestener,这里面封装的都是Activity中复杂的业务逻辑* @author 白乾涛*/public class UserLoginPresenter {private IUserBiz userBiz;private IUserLoginView userLoginView;private Handler mHandler = new Handler() {public void handleMessage(android.os.Message msg) {};};//在构造Prestener时,需要传递一个IUserLoginView接口的实例(其实就是Activity),只有这样,Prestener才能对此实例进行操作public UserLoginPresenter(IUserLoginView userLoginView) {this.userBiz = new UserBiz();this.userLoginView = userLoginView;}public void login() {userLoginView.showLoading();//为了更具扩展性及进一步解耦,具体的登录逻辑又通过同样的方式转移到了IUserBiz接口上(统一放在biz包下)userBiz.login(userLoginView.getUserName(), userLoginView.getPassword(), new OnLoginListener() {@Overridepublic void loginSuccess(final User user) {//需要再UI线程执行mHandler.post(new Runnable() {@Overridepublic void run() {userLoginView.toMainActivity(user);userLoginView.hideLoading();}});}@Overridepublic void loginFailed() {//需要在UI线程执行mHandler.post(new Runnable() {@Overridepublic void run() {userLoginView.showFailedError();userLoginView.hideLoading();}});}});}public void clear() {userLoginView.clearPassword();userLoginView.clearUserName();}}
bean包
/*** 一个业务bean,所有需要被操作的基本元素(不包括View)都应该定义在这里* @author 白乾涛*/public class User {private String username;private String password;public String getUsername() {return username;}public String getPassword() {return password;}public void setUsername(String username) {this.username = username;}public void setPassword(String password) {this.password = password;}}
biz(业务)包
/*** 复杂的业务逻辑在这里定义:如,登录、忘记密码、联系客服* @author 白乾涛*/public interface IUserBiz {public void login(String username, String password, OnLoginListener onLoginListener);}/*** 复杂业务逻辑的具体实现:如具体的登录代码在这里编写* 这里面有一个回调* @author 白乾涛*/public class UserBiz implements IUserBiz {@Overridepublic void login(final String username, final String password, final OnLoginListener onLoginListener) {new Thread() {@Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}if ("username".equals(username) && "password".equals(password)) {User user = new User();user.setUsername(username);user.setPassword(password);//成功时的回调onLoginListener.loginSuccess(user);} else {//失败时的回调onLoginListener.loginFailed();}}}.start();}}
/*** 这个其实不算是MVP中的东西(但一般都会有),这是一个回调,当执行复杂逻辑的同时做一些回调处理* @author 白乾涛*/public interface OnLoginListener {public void loginSuccess(User user);public void loginFailed();}
MVP模式 详解 案例的更多相关文章
- Javascript设计模式之装饰者模式详解篇
一.前言: 装饰者模式(Decorator Pattern):在不改变原类和继承的情况下动态扩展对象功能,通过包装一个对象来实现一个新的具有原对象相同接口的新的对象. 装饰者模式的特点: 1. 在不改 ...
- Extjs MVC开发模式详解
Extjs MVC开发模式详解 在JS的开发过程中,大规模的JS脚本难以组织和维护,这一直是困扰前端开发人员的头等问题.Extjs为了解决这种问题,在Extjs 4.x版本中引入了MVC开发模式, ...
- JavaScript严格模式详解
转载自阮一峰的博客 Javascript 严格模式详解 作者: 阮一峰 一.概述 除了正常运行模式,ECMAscript 5添加了第二种运行模式:"严格模式"(strict m ...
- HTTP协议头部与Keep-Alive模式详解
HTTP协议头部与Keep-Alive模式详解 .什么是Keep-Alive模式? 我们知道HTTP协议采用“请求-应答”模式,当使用普通模式,即非KeepAlive模式时,每个请求/应答客户和服务器 ...
- (" use strict")Javascript 严格模式详解
Javascript 严格模式详解 转载别人的博客内容,浏览了一遍,没有全部吸收,先保存一下链接 http://www.ruanyifeng.com/blog/2013/01/javascript_s ...
- HTTP协议Keep-Alive模式详解
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytp22 HTTP协议Keep-Alive模式详解 1.什么是Keep-Aliv ...
- Java开源生鲜电商平台-盈利模式详解(源码可下载)
Java开源生鲜电商平台-盈利模式详解(源码可下载) 该平台提供一个联合买家与卖家的一个平台.(类似淘宝购物,这里指的是食材的购买.) 平台有以下的盈利模式:(类似的平台有美菜网,食材网等) 1. 订 ...
- ext.js的mvc开发模式详解
ext.js的mvc开发模式详解和环境配置 在JS的开发过程中,大规模的JS脚本难以组织和维护,这一直是困扰前端开发人员的头等问题.Extjs为了解决这种问题,在Extjs 4.x版本中引入了MVC开 ...
- Docker Kubernetes Service 网络服务代理模式详解
Docker Kubernetes Service 网络服务代理模式详解 Service service是实现kubernetes网络通信的一个服务 主要功能:负载均衡.网络规则分布到具体pod 注 ...
随机推荐
- 【USACO 2.2.2】集合
[题目描述] 对于从1到N (1 <= N <= 39) 的连续整数集合,能划分成两个子集合,且保证每个集合的数字和是相等的.举个例子,如果N=3,对于{1,2,3}能划分成两个子集合,每 ...
- Mysql 数据类型使用说明
FLOAT 和DOUBLE 类型支持使用标准的浮点运算进行近似计算. DECIMAL类型用于存储精确的小数. 因为cpu不支持对DECIMAL的直接计算,所以在Mysql5.0及更高的版本中,MYSQ ...
- Linux进程或线程绑定到CPU
Linux进程或线程绑定到CPU 为了让程序拥有更好的性能,有时候需要将进程或线程绑定到特定的CPU,这样可以减少调度的开销和保护关键进程或线程. 进程绑定到CPU Linux提供一个接口,可以将进程 ...
- HTML5 QQ登录背景动态图片
预览效果如图所示: 代码如下: <!DOCTYPE html> <head> <meta http-equiv="Content-Type" cont ...
- z-index的理解 z-index 属性仅在节点的 position 属性为 relative, absolute 或者 fixed 时生效.
今天做游戏的Exercise模式的时候,发现把所有的div设置为position:absolute;后,点击play进入到游戏界面的时候,鼠标点击数字的时候,完全没反应.经过我的反复检查,发现只要给所 ...
- 浅谈Android序列化
序列化原因 序列化的原因基本可以归纳为以下三种情况: 永久性保存对象,保存对象的字节序列到本地文件中: 对象在网络中传递: 对象在IPC间传递. --- --- 序列化方法 在Android系统中关于 ...
- django自带User管理中添加自己的字段方法
#coding=utf-8 from django.db import models from django.contrib.auth.models import User, make_passwor ...
- [Struts2学习笔记] -- 自定义类型转换
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"% ...
- IOS 多个UIImageView 加载高清大图时内存管理
IOS 多个UIImageView 加载高清大图时内存管理 时间:2014-08-27 10:47 浏览:59人 当我们在某一个View多个UIImageView,且UIImageView都显示的是 ...
- C++进阶阅读
推荐的阅读顺序:level 1从<<essential c++>>开始,短小精悍,可以对c++能进一步了解其特性以<<c++ primer>>作字典和课 ...

