前言

写Android:如何编写“万能”的Activity的这篇文章到现在已经好久了,但是由于最近事情较多,写重构篇的计划就一直被无情的耽搁下来了,借这几天还算有点空余时间,把自己这桩心事了解下。
其实大家也知道Android:如何编写“万能”的Activity的这篇文章只是个引子,其实我真正想引出的是mvp设计模式,因为最近自己最近在用mvp做项目,自己对mvp有一些感悟,因此我将用mvp进行“万能”activity的重构。
同时也有一些朋友与我交流mvp,他们会被mvp中的m,v,p都应该放什么逻辑而困惑?会被mvp中的m应该怎么写而困惑?会被一个界面,怎么按照mvp来进行重构感到困惑?会被listview的adapter应该放在m层还是p层困惑?
我希望通过本文的讲解能帮助大家对mvp有一个更深的了解,现在进入主题内容。
正文内容
本文内容分为2部分:
第一部分会深入了解mvp到底是什么,它的好处等知识。第二部分会讲解如何利用mvp来重构“万能”Activity。
带你了解mvp
任何软件都是以数据为中心,为了能与用户进行交互,就需要提供界面支持用户对数据进行增删改查操作。
不管是mvc,mvp还是mvvm始终都在做一件事情:怎么样能更好的解决数据与界面之间的关系,以达到数据与界面之间的耦合更低,代码的复用性更高,代码的可测性更好。
本文的重点是讲解mvp,因此让我们开始了解下mvp是怎么组织数据与界面之间的关系的。我们先从mvp的结构图说起。
mvp的面容
网上有些关于mvp的结构图基本是以下样子
mvp结构图.png
我觉得这张图是有问题的,问题在于presenter把请求转交给model,model应该把处理结果返回给presenter,这张图是没有反映这个过程的。
正确的mvp的结构图是这样子的
mvp结构
我们先看下能从这张图中得到哪些信息?
看了mvp的整体结构图,我们以从底层到上层的顺序依次来介绍model,presenter,view。
model
先说下一些关于model的错误理解:
关于model的正确理解我们会在文中看到。
数据加工处理厂
通过应用mvp后的感受,我个人的感觉model是最难写的一层,并且也是最难懂的,因为model是整个应用或界面的数据加工处理厂,所谓数据加工厂就是对数据的获取,数据的解析,数据的存储,数据的分发,数据的增删改查等操作。意思就是凡是涉及到数据操作都是在model进行的,所以model不仅仅只是实体类的集合,同时还包含关于数据的各种处理操作。
三种数据源
数据的数据源有三种:内存,磁盘(文件或数据库等),网络。为了提升app的性能,有必要把经常访问的数据临时存入内存中;同时也为了提升app性能和为用户省流量省电,有必要把数据存入磁盘中;还有的数据是有必要从网络读取的。三个数据源不一定同时存在,比如不与网络交互的app,不存在网络数据源。所以凡是涉及到关于数据发生于三个数据源加工处理的操作的代码都要放在model中。
model为上层提供的服务
model从黑盒的角度来看为上层(指依赖于model的层比如present)提供的服务无非就2种:model为上层提供数据,model处理上层传递的数据
model为上层提供数据
上层会从model中去数据,那model会从三数据源中取数据,取的顺序是
上面的取数据过程是最简单的情况,复杂些还会涉及到从内存或磁盘中取到的数据是否过期,过期的话就应该从网络获取。从网络取得数据后需要把内存或磁盘的数据更新。
model处理上层传递的数据
model接收到上层传递的数据后,model会依次把数据扔给三个数据源去处理,有可能三个数据源都会处理数据,有可能只是其中一个处理,model会把处理的结果返回。
所以model会把解析好的数据提供给上层,上层对于数据的来源完全是透明的,上层完全不需要关心数据到底是来自内存,还是磁盘甚至是网络。同理上层只需要的把数据扔给model,上层唯一做的事情就是愉快的等待处理结果。
tip
mvc中的model是要和view进行交互的,而mvp中的model不会知道任何view的细节。
model中的所有操作都发生于普通线程。
关于model的介绍先到此,我们在来看下presenter。
presenter
presenter翻译成汉语的意思是主持人,提出者。从它的意思可以看出它有控制全场的作用。首先presenter是处于mvp的中间层,在view和model中起一个承上启下的作用。presenter会把view交给自己的命令进行一定的校验等操作交给model处理,会把model处理的结果交给view。
presenter封装业务
presenter不仅起一个桥梁的作用,它还会把业务逻辑代码给包揽下来。这样就可以减轻Activity的负担了,让Activity全心全意做它的view工作。那估计就有朋友犯迷糊了,哪些代码属于业务逻辑呢?比如一些校验代码。或者可以这样想只要是不属于view和model的代码基本都可以放在presenter中。
presenter负责刷新view
mvc或以前的关于view的写法一般都是这样,view在接收到数据后,自己来进行view的刷新或其他操作。但是mvp中presenter负责对view进行刷新,比如从model获取的数据,presenter会根据获取的数据成功与否来通知view应该是显示成功界面还是失败界面。这样就让Activity变的更轻了,变成了听别人指挥的傻白甜了。这时候的presenter就有点主持人,掌控者的味道了。
presenter持有的线程
Android中view的操作需要在ui线程里执行,其他耗时操作需要在普通线程执行。presenter会持有这2种线程:ui线程,普通线程。刷新view时,它切换为ui线程进行刷新,从model取数据切换为普通线程。假如使用rxjava的话,就特别简单了关于线程切换的事情。
tip
presenter从model中获取的数据就是解析好的数据,不需要出现解析数据的代码。
接着我们来看下view。
view
view层就很好理解了,就是用户直接看到的界面,mvp中的view是很省心的,比如更新view,接收数据。这些操作它都不需要操心,也不需要知道数据到底来自哪里,给我啥我显示啥就可以了。
一个view可以同时拥有多个presenter,也可以只有一个presenter。
Android中的Activity,Fragment在mvp中是作为view来使用的,这些Activity,Fragment的责任就小了,只关心界面相关的事情足矣。
各种Adapter是放在view层的。
总结
我们初步认识了mvp,mvp中的model,present,view到底是什么,他们之间的关系是什么样的,这只是初步认识mvp,关于mvp中还有很多细节需要介绍,比如android clean architecture 中model和presenter之间多了一层interactor,多的这层interactor是用来做什么的,model层是怎么架构的。google mvpmodel层要比android clean architecture 简单等,希望能在我后面的章节看到相关关于每层的详细介绍。我们开始进入我们的重构"万能"Activity的部分。
使用mvp设计模式对"万能"Activity进行重构
回忆下“万能”Activity的样子
我在上篇文章的“万能”的LoginActivity基础上增加了登录对话框的功能,“万能”LoginActivity的代码如下:
public LoginActivity extends Activity{
private EditText mUserNameView, mPasswordView;
private Button mLoginView;
public void initViews(){
.......
各种findViewById.....代码
//给登陆按钮加监听器
mLoginView.OnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String userName = mUserNameView.getText();
String password = mPasswordView.getText();
//验证用户输入的密码是否合法
if(!validate(userName) || !validate(password)){
告诉用户输入的用户名或密码不合法
} else{
//开始登陆
login(userName,password);
}
}
});
}
//登陆方法,用伪代码来写下网络请求
private void login(String userName,String password){
//增加登录进度对话框给用户友好用户体验
显示登录进度对话框...
HttpClient.getInstance().login(userName,password,
new ResponseListener(){
public void failed(Failed failed){
把登录进度对话框消失...
做失败相关的处理工作,比如给用户提示
把密码输入框清空,还比如登陆次数限制等
}
public void success(Response response){
把登录进度对话框消失...
做成功相关的处理工作
//暂且把用户信息的类叫做UserInfo,从json中解析数据,假设response.getContent()存在
String jsonContent = response.getContent();
JsonObject jsonObject = new JsonObject(jsonContent);
UserInfo userInfo = new UserInfo();
userInfo.name = jsonObject.optString("name");
userInfo.userId = jsonObject.optString("userId");
其他字段的解析......
//保存userInfo信息到数据表中,假设userDatabase已经存在
userDatabase.save(userInfo);
跳到app的主页
}
});
}
//验证给定的字符串是否合法,true 合法,false 不合法
private boolean validate(String str){
}
}
我们回忆了“万能”LoginActivity的代码后,开始重构。
开始重构
model
在使用mvp时,我一般有个习惯就是首先从model->presenter->view的顺序写代码,所以重构“万能”LoginActivity也先从model开始。前半部分关于model介绍过,model从黑盒的角度来说只有2个功能:一个是输出数据,一个是输入数据。因此登录中presenter只需要把账号,密码交给model,presenter唯一做的事情就是监听登录状态即可。model会把presenter传递的账号,密码交给服务器,model在把服务器返回的数据进行解析,存储在磁盘或内存中,把解析好的数据传递给presenter。那我们看下伪代码:
//管理登录的类,它是单例的,这就不写单例方法了
public class LoginManager{
//登录的监听器
public static interface LoginListener{
//登录成功
void onSuccessLogin(UserEntity user);
//登录失败
void onFailedLogin(Failed failed);
}
//登录方法
public void login(String name,String password,final LoginListener loginListener){
//假设HttpClient是一个请求网络服务的类
HttpClient.getInstance().login(userName,password,
new ResponseListener(){
public void failed(Failed failed){ loginListener.onFailedLogin(failed);
}
public void success(Response response){
//假设UserParse类已经存在,主要用来从response中解析UserEntity
UserEntity userEntity = UserParse(response.getContent());
//假设userDatabase是数据库存储类
userDatabase.store(userEntity);
//还可以把userEntity存入内存中,这得根据业务需求进行处理
loginListener.onSuccessLogin(userEntity);
}
});
}
}
登录的model层我们没有做的那么复杂,比如把服务器返回的用户信息存储在内存中,把服务器返回的token存储在磁盘中,实现自动登录功能等,本例子只是一个特别简单的登录功能,实际应用中登录需要考虑很多的东西,登录的modle层到此重构完毕。
presenter
上文中提到过presenter,presenter起连接view与model的作用,presenter封装业务作用,presenter有负责刷新view的作用。
我们梳理下presenter都应该包含哪些功能:
那让我们开始写代码,presenter层的类组织结构我是参照google mvp的presenter类组织结构来进行的,因为我认为google mvp presenter类结构更清晰,看下伪代码:
//登录的条约接口,分别定义了登录view的一些方法,和登录presenter的一些方法
public interface LoginContract{
//需要view层来实现的登录view接口,IView是所有view的基类
interface ILoginView extends IView{
void onShowSuccessLoginView(UserInfo userInfo);
void onShowFailedLoginView(int failed);
void showLoginingView();
void dissLoginingView();
}
//定义了登录presenter的一些方法,IPresenter是所有Presenter的基类
interface ILoginPresenter extends IPresenter<ILoginView>{
void login(String name,String password);
}
}
public interface IView{
void initView();
}
//IPresenter提供了一些基础方法,其实这些方法是对应Activity或Fragment的生命周期方法
public interface IPresenter<V extends IVew>{
void onStop();
void onResume();
void onDestroy();
void onPause();
void onStart();
void init(V view);
}
//登录的presenter
public class LoginPresenter implements ILoginPresenter{
private ILoginView mLoginView;
private LoginManager mLoginManager = LoginManager.getInstance();
public void init(ILoginView loginView){
mLoginView = loginView;
mLoginView.initView();
}
public void login(String name,String password){
//验证name,password的合法性,
if(validate(name) && validate(password)){
//假设NormalThread.exe方法可以让操作在普通线程里执行
mLoginView.showLoginingView();
NormalThread.exe(new Runnable(){
public void run(){
mLoginManager.login(name,password,
new LoginListener(){
public void onSuccessLogin(UserEntity userEntity){
//UserMapper类,负责把底层的UserEntity转化为view层使用的UserInfo UserInfo userInfo = UserMapper.map(userEntity);
//下面的代码在ui线程中执行,这就不写具体的实现了
mLoginView.onShowSuccessLoginView(userInfo);
mLoginView.dissLoginingView();
}
public void onFailedLogin(Failed failed){
//下面的代码在ui线程中执行,这就不写具体的实现了
mLoginView.onShowFailedLoginView(failed.failedState);
mLoginView.dissLoginingView();
}
});
}
}
}else{
//假设1代表账号,密码不合法
mLoginView.onShowFailedLoginView(1);
}
}
}
以上登录的Presenter层的伪代码都是关键代码,让我们看下以上代码都做了什么?
view
view层就很简单了,只是需要把基础设施建立好,直接看伪代码:
public abstract class BaseActivity extends FragmentActivity{
private Set<IPresenter> mAllPresenters = new HashSet<IPresenter>(1);
/** * 获取layout的id,具体由子类实现
* @return
*/
protected abstract int getLayoutResId();
/**
*需要子类来实现,获取子类的IPresenter,一个activity有可能有多个IPresenter
*/
protected abstract IPresenter[] getPresenters();
//初始化presenters,
protected abstract void onInitPresenters();
/** * 从intent中解析数据,具体子类来实现
* @param argIntent
*/
protected void parseArgumentsFromIntent(Intent argIntent){
}
private void addPresenters(){
IPresenter[] presenters = getPresenters();
if(presenters != null){
for(int i = 0; i < presenters.length; i++){
mAllPresenters.add(presneters[i]);
}
}
}
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(getLayoutResId());
if(getIntent() != null){
parseArgumentsFromIntent(getIntent());
}
addPresenters();
onInitPresents();
}
@Override
protected void onResume() {
super.onResume();
//依次调用IPresenter的onResume方法
for (IPresenter presenter:mAllPresenters ) {
if(presenter != null){
presenter.onResume();
}
}
}
...其他生命周期方法也是类似,调用IPresenter中相应的生命周期方法...
}
基础设施已经ok了,这时候我们就该重构"万能“LoginActivity了。
public class LoginActivity extends BaseActivity implements LoginConstract.ILoginView{
private LoginPresenter mLoginPresenter = new LoginPresenter();
protected int getLayoutResId(){
return R.layout.activity_login;
}
protected IPresenter[] getPresenters(){
return new IPresneter[]{ mLoginPresenter};
}
//初始化presenters,
protected void onInitPresenters(){
mLoginPresenter.init(this);
}
public void initView(){
...初始化view的代码...
//
mLoginButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
mLoginPresenter.login(name,password);
}
});
}
public void onShowSuccessLoginView(UserInfo userInfo){
....显示登录成功界面....
}
public void onShowFailedLoginView(int failed){
...显示登录失败界面...
}
public void showLoginingView(){
...显示登录进度条对话框...
}
public void dissLoginingView(){
...消失登录进度条对话框...
}
}
我们着重说下基础设施BaseActivity:
因为一个Activity是有可能包含多个Presenter的,所以需要在BaseActivity中是有必要把这些Presenter收集起来。
需要在BaseActivity的生命周期方法里面调用每个Presenter的相应周期方法。
重构以后的LoginActivity是不是很清爽,只保留与view相关的逻辑。
小结重构后的类结构
重构“万能”LoginActivity到此结束,我们小结下重构后的类结构:
重构感悟
疑惑
估计会有细心的朋友发现model层是LoginManager而不是像android clean architecture 中model层使用了respository,我个人觉得model层也没必要这么严格的按respository的架构方式来组织类结构,因为本例中登录功能实在是太简单了,所以就用最简单的一个LoginManager类来供上层调用。
优缺点
使用mvp重构“万能”Activity以后,带来了以下好处:
但同时也带来了一些缺点,比如创建的类多了很多,管理这些类的成本会增加。但是万事万物都有两面性,就看利与弊的大小了。我个人觉得mvp的利肯定是大于弊的,所以有必要采用这种架构来设计你的app。
提高生产效率
以上伪代码中我没有使用rxjava,dagger2,假如把它们应用于mvp中会让你事半功倍,rxjava可以让你在写presenter层和model层时,可以让presenter与model交互更简单,可以是model层变的尤为的简单比如从三大数据源取数据操作,我们自己用代码实现是可以的但是毕竟要花很多时间,但是用rxjava的一些操作符很容易做到。dagger2可以更好的帮助你进行依赖注入, 还有鼎鼎有名的retrofit也是可以提高效率的。
总结
本文我们了解了mvp以及每一层,以及使用mvp来重构“万能”Activity,其实每一层需要注意的东西还有很多,比如model层是最难写的一层。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!
# android
# activity
# Android:如何编写“万能”的Activity
# Android中fragment与activity之间的交互(两种实现方式)
# Android activity堆栈及管理实例详解
# Android实现Activity水平和垂直滚动条的方法
# Android开发中Activity创建跳转及传值的方法
# Android实现在子线程中更新Activity中UI的方法
# 详解Android开发中Activity的四种launchMode
# Android开发中关于获取当前Activity的一些思考
# Android判断Activity是否在最上层的方法
# Android编程中activity启动时出现白屏、黑屏问题的解决方法
# 重构
# 放在
# 对话框
# 会把
# 结构图
# 是有
# 子类
# 是一个
# 到此
# 让我们
# 多个
# 有可能
# 要把
# 这就
# 更高
# 有必要
# 都是
# 涉及到
# 提供给
# 来实现
相关文章:
网页设计与网站制作内容,怎样注册网站?
如何在IIS中配置站点IP、端口及主机头?
魔方云NAT建站如何实现端口转发?
如何通过多用户协作模板快速搭建高效企业网站?
高端建站三要素:定制模板、企业官网与响应式设计优化
表情包在线制作网站免费,表情包怎么弄?
为什么Go需要go mod文件_Go go mod文件作用说明
建站主机如何选?高性价比方案全解析
网站制作与设计教程,如何制作一个企业网站,建设网站的基本步骤有哪些?
如何在云虚拟主机上快速搭建个人网站?
宝塔建站助手安装配置与建站模板使用全流程解析
存储型VPS适合搭建中小型网站吗?
网站视频怎么制作,哪个网站可以免费收看好莱坞经典大片?
广德云建站网站建设方案与建站流程优化指南
代购小票制作网站有哪些,购物小票的简要说明?
javascript基本数据类型及类型检测常用方法小结
北京制作网站的公司排名,北京三快科技有限公司是做什么?北京三快科技?
如何配置FTP站点权限与安全设置?
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
南阳网站制作公司推荐,小学电子版试卷去哪里找资源好?
洛阳网站制作公司有哪些,洛阳的招聘网站都有哪些?
微信h5制作网站有哪些,免费微信H5页面制作工具?
盐城做公司网站,江苏电子版退休证办理流程?
如何选择CMS系统实现快速建站与SEO优化?
网站制作软件有哪些,制图软件有哪些?
香港服务器部署网站为何提示未备案?
台州网站建设制作公司,浙江手机无犯罪记录证明怎么开?
建站之星如何实现五合一智能建站与营销推广?
,巨量百应是干嘛的?
如何在七牛云存储上搭建网站并设置自定义域名?
c++ stringstream用法详解_c++字符串与数字转换利器
如何通过FTP服务器快速搭建网站?
网站制作大概要多少钱一个,做一个平台网站大概多少钱?
网站制作的方法有哪些,如何将自己制作的网站发布到网上?
深圳网站制作案例,网页的相关名词有哪些?
网站制作中优化长尾关键字挖掘的技巧,建一个视频网站需要多少钱?
智能起名网站制作软件有哪些,制作logo的软件?
制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?
视频网站制作教程,怎么样制作优酷网的小视频?
C++中引用和指针有什么区别?(代码说明)
阿里云网站制作公司,阿里云快速搭建网站好用吗?
如何高效完成自助建站业务培训?
如何在服务器上配置二级域名建站?
小说建站VPS选用指南:性能对比、配置优化与建站方案解析
如何选择可靠的免备案建站服务器?
如何在Golang中指定模块版本_使用go.mod控制版本号
广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?
如何选择美橙互联多站合一建站方案?
网站制作免费,什么网站能看正片电影?
音响网站制作视频教程,隆霸音响官方网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。