[译]Google官方关于Android架构中MVP模式的示例
概述
该示例(TODO-MVP)是后续各种示例演变的基础,它主要演示了在不带架构性框架的情况下实现M-V-P模式。其采用手动依赖注入的方式来提供本地数据源和远程数据源仓库。异步任务通过回调处理。

注意:MVP中View的概念是有所不同的:
- android.view.View类我们叫它 "Android View"
- MVP中从P接收命令的东西我们叫它”View”,后文说了,就是Fragment
Fragment
采用Fragment作为视图的载体有两个原因:
- Activity 和 Fragment分离恰好适合用来实现MVP:Activity是创建和连接View与Presenter的总控制器
- 适配平板设备的布局或多视图屏幕的场景都可利用Fragment框架的优势
关键概念
该App(指官方TODO-MVP示例)中有四个功能:
- Tasks
- TaskDetail
- AddEditTask
- Statistics
每个功能拥有下列代码分工:
- 定义VIew和Presenter之间的“契约”接口
- 一个Activity类负责创建Fragment和Presenter
- 一个Fragment类实现VIew接口
- 一个Presenter类实现Presenter接口
 一般来说,业务逻辑存在于Presenter中并依赖View完成Android中UI相关的工作。
View中几乎不包含逻辑:它将Presenter的命令转换为UI动作,并监听用户动作传递给Presenter。
“契约”接口用于定义View和Presenter之间的联系。
(译注:
有人立马就问到M哪去了,官方实例里M是轻的,与Repository和DataSource概念分离。在官方的另一个示例中有进一步的描述:http://www.cnblogs.com/x3d/p/6108998.html
/**
 * Immutable model class for a Task.
 */
public final class Task {
    @NonNull
    private final String mId;
    @Nullable
    private final String mTitle;
    @Nullable
    private final String mDescription;
    private final boolean mCompleted;
@NonNull
    public String getId() {
        return mId;
    }
    @Nullable
    public String getTitle() {
        return mTitle;
    }
    @Nullable
    public String getTitleForList() {
        if (!Strings.isNullOrEmpty(mTitle)) {
            return mTitle;
        } else {
            return mDescription;
        }
    }
    @Nullable
    public String getDescription() {
        return mDescription;
    }
    public boolean isCompleted() {
        return mCompleted;
    }
    public boolean isActive() {
        return !mCompleted;
    }
    public boolean isEmpty() {
        return Strings.isNullOrEmpty(mTitle) &&
               Strings.isNullOrEmpty(mDescription);
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Task task = (Task) o;
        return Objects.equal(mId, task.mId) &&
               Objects.equal(mTitle, task.mTitle) &&
               Objects.equal(mDescription, task.mDescription);
    }
    @Override
    public int hashCode() {
        return Objects.hashCode(mId, mTitle, mDescription);
    }
    @Override
    public String toString() {
        return "Task with title " + mTitle;
    }
}
DataSource、Repository相关代码布局:

契约接口定义:
package com.example.android.architecture.blueprints.todoapp.taskdetail;
import com.example.android.architecture.blueprints.todoapp.BasePresenter;
import com.example.android.architecture.blueprints.todoapp.BaseView;
/**
 * This specifies the contract between the view and the presenter.
 */
public interface TaskDetailContract {
    interface View extends BaseView<Presenter> {
        void setLoadingIndicator(boolean active);
        void showMissingTask();
        void hideTitle();
        void showTitle(String title);
        void hideDescription();
        void showDescription(String description);
        void showCompletionStatus(boolean complete);
        void showEditTask(String taskId);
        void showTaskDeleted();
        void showTaskMarkedComplete();
        void showTaskMarkedActive();
        boolean isActive();
    }
    interface Presenter extends BasePresenter {
        void editTask();
        void deleteTask();
        void completeTask();
        void activateTask();
    }
}
)
项目依赖
- Common Android support 库 (com.android.support.*)
- Android Testing Support 库 (Espresso, AndroidJUnitRunner…)
- Mockito
- Guava (null checking)
特性
复杂度 - 易理解性
使用架构性框架/库/工具类:
无
概念复杂度
低,基于Android的纯MVP实现。
可测试性
单元测试
高,Presenter可被单元测试,和仓库与数据源一样。
UI 测试
高,允许使用fake模块注入伪造(fake)数据进行测试。
代码对比
相比于不使用架构的传统项目,该示例引入了额外的类和接口:Presenter、Repositiy、契约接口等,所以MVP中代码行数更多一些。
| 语言 | 文件 | 空行 | 注释 | 代码 | 
|---|---|---|---|---|
| Java | 46 | 1075 | 1451 | 3451 | 
| XML | 34 | 97 | 337 | 601 | 
| SUM | 80 | 1172 | 1788 | 4052 | 
可维护性
简化功能的维护或新功能的迭代
高
学习成本
低
相关功能代码能容易查找定位、代码职责清晰。开发者无需熟悉任何外部的依赖就能对该项目开展工作。
Github项目地址:https://github.com/googlesamples/android-architecture/
[译]Google官方关于Android架构中MVP模式的示例的更多相关文章
- Google官方关于Android架构中MVP模式的示例续-DataBinding
		基于前面的TODO示例,使用Data Binding库来显示数据并绑定UI元素的响应动作. 这个示例并未严格遵循 Model-View-ViewModel 或 Model-View-Presenter ... 
- 浅谈Android架构之MVP,MVVM
		概述 MVP(Model-View-Presenter)是传统MVC(Model-View-Controller)在Android开发上的一种变种.进化模式.主要用来隔离UI.UI逻辑和业务逻辑.数据 ... 
- Android Studio重构之路,我们重新来了解一下Google官方的Android开发工具
		Android Studio重构之路,我们重新来了解一下Google官方的Android开发工具 记得我的第一篇博客就是写Android Studio,但是现在看来还是有些粗糙了,所有重构了一下思路, ... 
- 下载Google官方/CM Android源码自己主动又一次開始的Shell脚本
		国内因为某种原因,下载CM或Google官方的Android源码总easy中断.总看着机器.一中断就又一次运行repo sync还太麻烦,所以我特意编写了一段shell脚本(download.sh). ... 
- 如何结合整洁架构和MVP模式提升前端开发体验 - 整体架构篇
		本文不详细介绍什么是整洁架构以及 MVP 模式,自行查看文章结尾相关链接文章. 整洁架构粗略介绍 下图为整洁架构最原始的结构图: Entities/Models:实体层,官方说法就是封装了企业里最通用 ... 
- Android APP架构设计——MVP的使用示例
		0. 前言 为了更好地进行移动端架构设计,我们最常用的就是MVC.MVP和MVVM,作为三个最耳熟能详的三大架构,应用可谓非常广泛.对于这三种架构设计以及优缺点已经在Android APP架构设计-- ... 
- android中MVP模式(一) - 清风明月的专栏 - CSDN博客
		presenter 主持人.主导器 ====== 1. 明确需求,界面如下:可存,可根据id读取数据. 包结构图 2. 建立bean public class UserBean { private S ... 
- Android中MVP模式与MVC模式比較(含演示样例)
		原文链接 http://sparkyuan.me/ 转载请注明出处 MVP 介绍 MVP模式(Model-View-Presenter)是MVC模式的一个衍生. 主要目的是为了解耦,使项目易于维护. ... 
- Android应用中MVP开发模式
		所谓MVP(Model-View-Presenter)模式.是将APP的结构分为三层: view - UI显示层 view 层主要负责: 提供UI交互 在presenter的控制下修改UI. 将业务事 ... 
随机推荐
- UWP简单示例(三):快速开发2D游戏引擎
			准备 IDE:VisualStudio 2015 Language:VB.NET/C# 图形API:Win2D MSDN教程:UWP游戏开发 游戏开发涉及哪些技术? 游戏开发是一门复杂的艺术,编码方面 ... 
- 《如何使用Javascript判断浏览器终端设备》
			WEB开发中如何通过Javascript来判断终端为PC.IOS(iphone).Android呢? 可以通过判断浏览器的userAgent,用正则来判断手机是否是ios和Android客户端. va ... 
- CSS margin详解
			以下的分享是本人最近几天学习了margin知识后,大有启发,感觉以前对margin的了解简直太浅薄.所以写成以下文章,一是供自己整理思路:二是把知识分享出来,避免各位对margin属性的误解.内容可能 ... 
- 给缺少Python项目实战经验的人
			我们在学习过程中最容易犯的一个错误就是:看的多动手的少,特别是对于一些项目的开发学习就更少了! 没有一个完整的项目开发过程,是不会对整个开发流程以及理论知识有牢固的认知的,对于怎样将所学的理论知识应用 ... 
- 换个角度看微信小程序[推荐]
			去年参加几次技术沙龙时,我注意到一个有意思的现象:与之前大家统一接受的换名片不同,有些人并不愿意被添加微信好友--"不好意思,不熟的人不加微信". 这个现象之所以有意思,是因为名片 ... 
- .JavaWeb文件上传和FileUpload组件使用
			.JavaWeb文件上传 1.自定义上传 文件上传时的表单设计要符合文件提交的方式: 1.提交方式:post 2.表单中有文件上传的表单项:<input type="file" ... 
- “前.NET Core时代”如何实现跨平台代码重用 ——源文件重用
			微软在2002年推出了第一个版本的 .NET Framework,这是一个主要面向Windows 桌面(Windows Forms)和服务器(ASP.NET Web Forms)的基础框架.在此之后, ... 
- C#与yaml解析
			YAML 官方网站称 YAML 是"一种所有编程语言可用的友好的数据序列化标准".YAML Ain't Markup Language,和GNU一样,YAML是一个递归着说&quo ... 
- 浅析Go语言的Interface机制
			前几日一朋友在学GO,问了我一些interface机制的问题.试着解释发现自己也不是太清楚,所以今天下午特意查了资料和阅读GO的源码(基于go1.4),整理出了此文.如果有错误的地方还望指正. GO语 ... 
- [WinAPI] 获取窗口句柄的几种方法
			1.使用FindWindow函数获取窗口句柄 示例:使用FindWindow函数获取窗口句柄,然后获得窗口大小,并且移动窗口到指定位置. 我们想获得酷我音乐盒的窗口句柄并移动它,该怎么办呢? 首先打开 ... 
