Android MVP升级路(二)时尚版
前言
第一篇文章的结尾对时尚版MVP结构做了一个简单的预告,下面继续从时尚版MVP说起。
时尚版MVP架构-Model层的优化
在从乞丐版MVP架构优化成平民版MVP架构的过程中,几乎每个单元都做了很大优化并封装到了base层,但是唯独Model层没什么变化。所以,时尚版MVP架构的优化主要就是对Model层的优化。
单独封装,集中管理
Model层相比其他单元来说比较特殊,因为它们更像一个整体,只是单纯的帮上层拿数据而已。再就是MVP的理念是让业务逻辑互相独立,这就导致每个的网络请求也被独立成了单个Model,这种方式在实际开发中就会出现一些问题:
- 无法对所有Model统一管理。
- 每个Model对外提供的获取数据方法不一样,上层请求数据没有规范。
- 代码冗余高,网络数据请求除URL和参数外其他大概都一样的。
- 对已存在的Model管理困难,不能直观的统计已存在的Model。
所以我们更希望Model层是一个庞大且独立单一模块,请求方式规范化,管理Model更加直观。
时尚版MVP架构Model层结构
如上图所示,时尚版MVP架构的Model层中,Presenter 请求数据不再直接调用具体的Model对象,统一以 DataModel 类作为数据请求层的入口,以常量类 Token 区别具体请求。 DataModel会根据Token的不同拉取底层对应的具体Model。
优化之后的Model层是一个庞大而且独立的模块,对外提供统一的请求数据方法与请求规则,这样做的好处有很多:
- 数据请求单独编写,无需配合上层界面测试。
- 统一管理,修改方便。
- 利用Token类可以直观的统计出已存在的请求接口。
代码实现
根据上节结构图中的描述在考虑到实际情况,我们需要设计以下几个类:
-
DataModel
: 数据层顶级入口,项目中所有数据都由该类流入和流出,负责分发所有的请求数据。 -
Token
:数据请求标识类,定义了项目中所有的数据请求。 -
BaseModel
:所有Model的顶级父类,负责对外提供数据请求标准,对内为所有Model提供请求的底层支持。
最后实现后理想的请求数据方法是:
BaseModel
BaseModel中定义了对外的请求数据规则,包括设置参数的方法和设置Callback的方法,还可以定义一些通用的数据请求方法,比如说网络请求的Get和Post方法。
public abstract class BaseModel<T> {
//数据请求参数
protected String[] mParams;
/**
* 设置数据请求参数
* @param args 参数数组
*/
public BaseModel params(String... args){
mParams = args;
return this;
}
// 添加Callback并执行数据请求
// 具体的数据请求由子类实现
public abstract void execute(Callback<T> callback);
// 执行Get网络请求,此类看需求由自己选择写与不写
protected void requestGetAPI(String url,Callback<T> callback){
//这里写具体的网络请求
}
// 执行Post网络请求,此类看需求由自己选择写与不写
protected void requestPostAPI(String url, Map params,Callback<T> callback){
//这里写具体的网络请求
}
}
写好了BaseModel后再看实现具体Model的方法:
public class UserDataModel extends BaseModel<String> {
@Override
public void execute(final Callback<String> callback) {
// 模拟网络请求耗时操作
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
// mParams 是从父类得到的请求参数
switch (mParams[0]){
case "normal":
callback.onSuccess("根据参数"+mParams[0]+"的请求网络数据成功");
break;
case "failure":
callback.onFailure("请求失败:参数有误");
break;
case "error":
callback.onError();
break;
}
callback.onComplete();
}
},2000);
}
}
从上面代码段可以看出,实现具体的Model请求时必须要重写BaseModel的抽象方法execute
。
DataModel
由于DataModel负责数据请求的分发,所以最初打算作成一个简单工厂模式的样子,通过switch(token)
语句判断要调用的Model。
但如果这样设计的话,在实际开发中我们每次添加一个数据请求接口,不光需要新建对应的Model和Token,还需要在DataModel类的switch(token)
语句中新增加对应的判断,贼麻烦~
思来想去,我觉得利用反射机制会是一个比较理想的办法,请求数据时以具体Model的包名+类型作为Token,利用反射机制直接找到对应的Model。
public class DataModel {
public static BaseModel request(String token){
// 声明一个空的BaseModel
BaseModel model = null;
try {
//利用反射机制获得对应Model对象的引用
model = (BaseModel)Class.forName(token).newInstance();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
return model;
}
}
Token
由于上节中DataModel使用反射机制获取对应Model的引用,所以Token中存的就应该是对应Model的包名+类名:
public class Token {
// 包名
private static final String PACKAGE_NAME = "com.jesse.mvp.data.model.";
// 具体Model
public static final String API_USER_DATA = PACKAGE_NAME + "UserDataModel";
}
使用方式
完成了Model层之后再去Presenter调用数据时的样子就舒服多了:
DataModel
// 设置请求标识token
.request(Token.API_USER_DATA)
// 添加请求参数,没有则不添加
.params(userId)
// 注册监听回调
.execute(new Callback<String>() {
@Override
public void onSuccess(String data) {
//调用view接口显示数据
mView.showData(data);
}
@Override
public void onFailure(String msg) {
//调用view接口提示失败信息
mView.showFailureMessage(msg);
}
@Override
public void onError() {
//调用view接口提示请求异常
mView.showErrorMessage();
}
@Override
public void onComplete() {
// 隐藏正在加载进度条
mView.hideLoading();
}
});
添加model步骤
- 新建一个Model并继承BaseModel,完成具体的数据请求。
- 在Token中添加对用的Model包名+类名。注意写好注释,方便以后查阅。
总结
经过优化的Model层很好的统一化了请求方法规范,利用BaseModel不仅有效的减少了数据请求的冗余代码,最关键的还是得到了将所有Model的集中控制权,例如我们想给所有的请求都加上coockies,直接在BaseModel层做处理即可。
时尚版MVP虽然只对Model层进行了优化,实际开发中已经能发挥很大的作用。
下面一章旗舰版将三层同时优化。
- 关于ssh命令的几个使用小技巧(r11笔记第27天)
- Golang语言社区--消息触发服务器启动基础模块分享
- NYOJ------汉诺塔(一)
- HDUOJ----Coin Change
- Golang语言社区--列出目录和遍历目录的方法
- HDUOJ-------单词数
- insert导致的性能问题大排查(r11笔记第26天)
- NYOJ-----最少乘法次数
- nyOJ-----韩信点兵
- HDUOJ-----A == B ?
- 用Oracle的眼光来学习MySQL 5.7的sys(上)(r11笔记第24天)
- Golang下通过syscall调用win32的api
- NYOJ----蛇形填数
- Golang语言 syscall 例子
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- Linux 命令利用scp实现从服务器共享地址上传下载文件、文件夹实例演示,scp命令的参数详解
- Oracle 数据库利用sql语句判断某个表是否是临时表实例演示,达梦数据库查询出所有临时表
- JavaScript 技术篇-一段js代码展示可以随鼠标移动变换样式的卡通人物,动态女生眼睛跟着鼠转动
- PyQt5 图形界面-用Qt Designer来设计UI界面,并转化为python代码运行
- Python 技术篇-python生成html源码功能实现演示,html代码自动生成技巧。列表生成式的灵活应用。
- Python 技术篇-pyHook键盘鼠标监听事件,监测鼠标键盘按键。超简单,几行代码搞定。
- Python 技术篇-用mutagen库提取MP3歌曲图片
- Python 典藏篇-Microsoft Visual C++ 14.0 is required,官方vc++运行库工具一键式解决!
- Python 技术篇-邮件写入html代码,邮件发送表格,邮件发送超链接,邮件发送网络图片
- 面经手册 · 第11篇《StringBuilder 比 String 快?空嘴白牙的,证据呢!》
- domReady的理解
- Map集合排序
- Chrome 技术篇-一台电脑设置多个独立chrome方法。chrome独立多开技术。
- 023.Ubuntu常见个性化配置
- 快速学习-ElasticJob的FAQ