Android上实现MVP模式的途径
今天我想分享我在Android上实现MVP(Model-View-Presenter)模式的方法。如果你对MVP模式还不熟悉,或者不了解为什么要在Android应用中使用MVP模式,推荐你先阅读这篇维基百科文章和这篇博客。
使用Activity和Fragment作为View合适么?
目前,在很多使用了MVP模式的Android项目中,主流做法是将Activity和Fragment作为视图层来进行处理。而Presenters通常是通过继承被视图层实例化或者注入的对象来得到的。我认可这种方式可以节省掉那些让人厌烦的”import android.*”语句,并且将Presenters从Activity的生命周期中分离出来, 这使项目后续的维护会变得简便很多。但另一方面, Activity有一个很复杂的生命周期(Fragment的生命周期可能会更复杂)。而这些生命周期很有可能对项目的业务逻辑有非常重要的影响。Activity可以获取Context和各种Android系统服务。Activity可以发送Intent,启动Service和执行FragmentTransisitons等等。在我看来,这些错综复杂的方面不应该是视图层涉及的领域(视图的功能只是显示数据,从用户那里获取输入数据。在理想情况下,视图应该避免业务逻辑,无需单元测试)。基于上述原因,我对目前的主流做法并不赞同,所以我尝试使用Activity和Fragment作为Presenters。
使用Activity和Fragment作为Presenters
1、去除所有的view
将Activity和Fragment作为Presenter最大的困难就是如何将关于UI的逻辑分离出来。我的解决方案是:让需要作为Presenter的Activity或者Fragment来继承一个抽象的类。这样关于View各种组件的初始化以及逻辑,都可以在继承了抽象类的方法中进行操作。而当继承了该抽象类的class需要对某些组件进行操作的时候,只需要调用继承自抽象类的方法而不必考虑Presenter类型。在抽象类里面会有一个实例化的接口,这个接口里面的初始化方法就会对view进行实例化,这个接口我称为Vu,如下所示:
|
1
2
3
4
|
public interface Vu { void init(LayoutInflater inflater, ViewGroup container); View getView();} |
如你所见,Vu定义了一个通用的初始化例程,我可以通过它来传递一个填充器和一个容器视图。它也有一个方法可以获得一个View的实例,每一个presenter将会和它自己的Vu关联,这个presenter将会实现这个接口(直接或间接地去实现一个继承自Vu的接口)。
2、创建Presenter基类
现在我有了抽象的View的基础,我可以着手定义一个Activity或者Fragment基类来充分利用Vu从而实现View的实例化。我是通过利用普通类型和抽象方法来实现的,它定义了一个特殊的表示Presenter的Vu类。这是实现中最单调乏味的部分,因为我需要重新实现想要的相似逻辑或者每一个Presente基类。
下面是我实现的Activity例子:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
public abstract class BasePresenterActivity<V extends Vu> extends Activity { protected V vu; @Override protected final void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); try { vu = getVuClass().newInstance(); vu.init(getLayoutInflater(), null); setContentView(vu.getView()); onBindVu(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } } @Override protected final void onDestroy() { onDestroyVu(); vu = null; super.onDestroy(); } protected abstract Class<V> getVuClass(); protected void onBindVu(){}; protected void onDestroyVu() {};} |
下面是我实现的Fragment例子:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
public abstract class BasePresenterFragment<V extends Vu> extends Fragment { protected V vu; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); } @Override public final View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = null; try { vu = getVuClass().newInstance(); vu.init(inflater, container); onBindVu(); view = vu.getView(); } catch (java.lang.InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } return view; } @Override public final void onDestroyView() { onDestroyVu(); vu = null; super.onDestroyView(); } protected void onDestroyVu() {}; protected void onBindVu(){}; protected abstract Class<V> getVuClass();} |
相同的逻辑可以用在Activity和Fragment类型上,比如支持库中Activity和Fragment等等。
可以看到,我重写了创建视图view的方法(onCreate、onCreateView)和销毁视图view的方法(onDestroy、onDestroyView)。我选择重写这些方法目的是强制使用抽象实例Vu。一旦它们被重写,我就可以创建新的生命周期方法,来精确控制对其初始化和销毁,即onBindVu和onDestroyVu。这样做的好处就是,两种类型的presenter都可以利用同样的生命周期事件签名来实现。这也消除了Activity和Fragemnt生命周期差异的影响,使得两者之间的转换更加容易。 (你也可能会注意到,我并没有真正的利用InstantiationException 或者IllegalAccessException做一些异常处理。这仅仅是我比较懒罢了,因为如果我正确地使用这些类就不会抛出这些异常。)
3、写一个可以工作的例子
现在,我们可以使用刚才构建的框架。简单起见,我写一个“Hello World”的例子。我会从创建一个实现了Vu接口的类开始写:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public class HelloVu implements Vu { View view; TextView helloView; @Override public void init(LayoutInflater inflater, ViewGroup container) { view = inflater.inflate(R.layout.hello, container, false); helloView = (TextView) view.findViewById(R.id.hello); } @Override public View getView() { return view; } public void setHelloMessage(String msg){ helloView.setText(msg); }} |
下一步,我会创建一个Presenter来操作这个view:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public class HelloActivity extends BasePresenterActivity<HelloVu> { @Override protected void onBindVu() { vu.setHelloMessage("Hello World!"); } @Override protected Class<MainVu> getVuClass() { return HelloVu.class; }} |
等等……有耦合警告!
你可能注意到了,HelloVu类直接实现了Vu接口,Presenter的getVuClass()方法直接引用了实现类。常规的MVP模式中,Presenter要通过接口与他们的View解耦。当然,你也可以这么做。为了避免直接实现Vu接口,我们可以创建一个扩展了Vu的IHelloView接口,然后使用这个接口作为Presenter的泛型类型。那么Presenter看起来应该是这样的:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public class HelloActivity extends BasePresenterActivity<IHelloVu> { @Override protected void onBindVu() { vu.setHelloMessage("Hello World!"); } @Override protected Class<MainVu> getVuClass() { return HelloVuImpl.class; }} |
在我使用强大的模拟工具过程中,并没有看到一个接口下面实现Vu所带来的好处。但是对于我来说一个好的方面是,即使没有定义Vu接口它也能够工作,唯一的需求就是你最终还要实现Vu。
4、测试
通过以上几步我们可以发现,在去除了UI逻辑之后Activity变得非常简洁。同时,相关的测试也变的异常简单。请看如下单元测试:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
public class HelloActivityTest { HelloActivity activity; HelloVu vu; @Before public void setup() throws Exception { activity = new HelloActivity(); vu = Mockito.mock(HelloVu.class); activity.vu = vu; } <a href="http://www.jobbole.com/members/test/" rel="nofollow">@Test</a> public void testOnBindVu(){ activity.onBindVu(); verify(vu).setHelloMessage("Hello World!"); }} |
以上代码是一段标准的JUnit单元测试的代码,不需要在Android设备中部署运行。当然我们测试的Activity要足够简单。特殊情况下,在测试需要某些硬件支持的方法的时候,你可能需要使用Android设备。例如当你想测试Activity生命周期中的onResume()方法。在缺乏硬件设备支持环境的时候,super.onResume()会报错。还好我们可以使用一些工具,例如Robolectric、还有Android Studio 中的Gradle 1.1 插件中内置的testOptions { unitTests.returnDefaultValues = true }选项。此外,你仍然可以将这些生命周期按照下面的方式抽离出来:
|
1
2
3
4
5
6
7
8
9
10
11
|
... @Override protected final void onResume() { super.onResume(); afterResume(); } protected void afterResume(){}... |
现在,你可以把应用程序中特定的逻辑代码转移到生命周期事件中,并且在没有Android设备的情况下运行测试了。
意外收获:使用Adapter作为Presenter
将Activity作为Presenter已经足够巧妙了吧,如果是adapter,情况会更复杂。它们可以是View或者Presenter么?废话不多说,请看如下的代码:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
public abstract class BasePresenterAdapter<V extends Vu> extends BaseAdapter { protected V vu; @Override public final View getView(int position, View convertView, ViewGroup parent) { if(convertView == null) { LayoutInflater inflater = (LayoutInflater) parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE); try { vu = (V) getVuClass().newInstance(); vu.init(inflater, parent); convertView = vu.getView(); convertView.setTag(vu); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } } else { vu = (V) convertView.getTag(); } if(convertView!=null) { onBindListItemVu(position); } return convertView; } protected abstract void onBindListItemVu(int position); protected abstract Class<V> getVuClass();} |
正如你看到的,实现方式和Activity和Fragment的Presenter是一样的。然而,我不是用空的onBindVu方法,而是用参数为整型的position的onBindListItemVu方法。同时,我仍然沿用了View Holder模式。
总结和Demo项目
这篇文章介绍了一种实现MVP模式的方法。从中我发现唯一的途径就是网上寻找答案。我非常期待其他Android开发者的反馈,是否有人在用这个方法?你发现它有用么?我是否过于大胆(疯狂)?如果是的话,这是一个好办法吗?
我已经把这套方法(和一些其他的比如Dagger开源库)集成在一个开源框架上,并且即将公布。与此同时,我在Github上面有一个demo项目,望各位不吝赐教。
Android上实现MVP模式的途径的更多相关文章
- Android上的MVP:如何组织显示层的内容
MVP(Model View Presenter)模式是著名的MVC(Model View Controller)模式的一个演化版本,目前它在Android应用开发中越来越重要了,大家也都在讨论关于M ...
- Android 中的MVP 模式
MVP模式的核心思想: MVP把Activity中的UI逻辑抽象成View接口,把业务逻辑抽象成功接口,Model类还是原来的Model. MVC 其中View层其实就是程序的UI界面,用于向用户展示 ...
- [译]Google官方关于Android架构中MVP模式的示例
概述 该示例(TODO-MVP)是后续各种示例演变的基础,它主要演示了在不带架构性框架的情况下实现M-V-P模式.其采用手动依赖注入的方式来提供本地数据源和远程数据源仓库.异步任务通过回调处理. 注意 ...
- Google官方关于Android架构中MVP模式的示例续-DataBinding
基于前面的TODO示例,使用Data Binding库来显示数据并绑定UI元素的响应动作. 这个示例并未严格遵循 Model-View-ViewModel 或 Model-View-Presenter ...
- Android开发之MVP模式的使用
前几天发现,在Android项目代码里有一个Activity类行数居然有1000多行,而600行左右都是逻辑控制,真正和页面控件处理相关的代码不多,虽然可以用#region <>...#e ...
- 在Android上启用Kiosk模式
我们的云帆机器人(上面运行的安卓程序)有一个线下场景是商场,由于商场人多,总会遇到一些用户在我们的app里乱点,然后会跳出程序进入到系统设置的一些界面,这样很不友好. 比如程序中有一些需要输入文字的地 ...
- android中的MVP模式
1.建立bean public class UserBean { private String mFirstName; private String mLastName; public UserBea ...
- Android开发学习--MVP模式入门
1.模型与视图完全分离,我们可以修改视图而不影响模型2.可以更高效地使用模型,因为所有的交互都发生在一个地方——Presenter内部3.我们可以将一个Presenter用于多个视图,而不需要改变Pr ...
- Android开发MVP模式解析
http://www.cnblogs.com/bravestarrhu/archive/2012/05/02/2479461.html 在开发Android应用时,相信很多同学遇到和我一样的情况,虽然 ...
随机推荐
- Kafka:ZK+Kafka+Spark Streaming集群环境搭建(九)安装kafka_2.11-1.1.0
如何搭建配置centos虚拟机请参考<Kafka:ZK+Kafka+Spark Streaming集群环境搭建(一)VMW安装四台CentOS,并实现本机与它们能交互,虚拟机内部实现可以上网.& ...
- PHP 自定义方法实现数组合并
在PHP中提供了强大的数组功能,对于数组的合并也提供了两个方法:array_merge 和 array_merge_recursive 但对于我们千变万化的业务来说这些内置的方法并不完全能满足我们的要 ...
- Softmax 函数的特点和作用是什么?
作者:张欣链接:https://www.zhihu.com/question/23765351/answer/98897364来源:知乎著作权归作者所有,转载请联系作者获得授权. softmax 回归 ...
- artTemplate子模板include
art.Template:https://github.com/aui/art-template 下面来实现利用模版来实现递归调用生成tree <script type="text/h ...
- [Canvas]人物型英雄出现(前作仅为箭头)
源码点此下载,用浏览器打开index.html观看. 代码: <!DOCTYPE html> <html lang="utf-8"> <meta ht ...
- Android Activity 及其子类
本文内容 ListActivity TabActivity LauncherActivity ExpandableListActivity PerferenceActivity 这些类都继承 Acti ...
- 配置Oracle访问SQL地理数据库
Oracle访问空间数据 ArcSDE是ArcGIS的空间数据引擎,它是在关系数据库管理系统(RDBMS)中存储和管理多用户空间数据库的通路.以前连接方式有两种,服务连接与直接连接(简称"直 ...
- c数据库读写分离和负载均衡策略
最近在学习数据库的读写分离和主从复制,采用的是一主多从策略,采用轮询的方式,读取从数据库的内容.但是,假如某一台从数据库宕机了,而客户端不知道,每次轮选到此从数据库,不都要报错?到网上查阅了资料,找到 ...
- poj 3042 Grazing on the Run
这个题目原型应该是吃完所有的草丛的最小时间,现在变成了每个草丛被吃的时间和,貌似如果还是按照原来的dp方法dp[i][j]表示吃完i到j的草丛的花掉的时间的话,有两个因素会影响后面的决策,一个是花掉的 ...
- HTTP长连接与短链接
想要充分了解HTTP长连接,需要首先知道一些基本概念: TCP连接 当网络通信时采用TCP协议时,在真正的读写操作之前,server与client之间必须建立一个连接,当读写操作完成后,双方不再需要这 ...