Android 进行单元測试难在哪-part3
- 原文链接 : HOW TO MAKE OUR ANDROID APPS UNIT TESTABLE (PT. 1)
- 原文作者 : Matthew Dupree
- 译文出自 : 开发技术前线 www.devtf.cn
- 译者 : chaossss
- 校对者: tiiime
- 状态 : 完毕
在 Android 应用中进行单元測试非常困难。有时候甚至是不可能的。在之前的两篇博文中,我已经向大家解释了在 Android 中进行单元測试如此困难的原因。而上一篇博文我们通过分析得到的结论是:正是 Google 官方所提倡的应用架构方式使得在 Android 中进行单元測试变成一场灾难。由于在官方提倡的架构方式中,Google 似乎希望我们将业务逻辑都放在应用的组件类中(比如:Activity,Fragment。Service,等等……)。而这种开发方式也是我们一直以来使用的开发模板。
在这篇博文中。我列举出几种架构 Android 应用的方法,使用这些方法进行开发能让单元測试变得轻松些。但正如我在序中所说,我最推崇的办法始终是 Square 公布的博文:Square:从今天開始抛弃Fragment吧。 中所用的通用方法。由于这种方法是由 Square 中的 Android 开发project师想出来的。所以我会在接下来的博文中将这个办法叫作“Square 大法”。
Square 大法的核心思想是:把应用组件类中的业务逻辑所有移除(比如:Activity,Fragment,Service。等等……)。并且把业务逻辑转移到业务对象,而这些业务对象都是被依赖注入的纯 Java 对象。以及与 Android 无关的接口在此的 Android 特定实现。假设我们在开发应用的时候使用 Square 大法,那进行单元測试就简单多了。
在这篇博文中,我会解释 Square 大法是怎样帮助我们重构 UI 无关的应用组件(比如我们在之前的博文中讨论的 SessionCalendarService),并让对它进行单元測试变得easy很多。
用 Square 大法重构 UI 无关的应用组件
用 Square 大法重构类似于 Service,ContentProvider,BroadcastReceiver这种 UI 无关的应用组件相对来说比較easy。
我再说一次我们要做的事情吧:把在这些类中的业务逻辑移除。并把它们放到业务对象中。
由于“业务逻辑”是非常easy有歧义的词语,我来解释下我使用“业务逻辑”这个词时,它所代表的含义吧。当我提到“业务逻辑”,它的含义和维基百科上的解释是一致的:程序中依据现实世界中的规则用于决定数据将怎样被创建,展示,储存和改动的那部分代码。那么如今我们就能够就“业务逻辑”这个词的含义达成共识了,那就来看看 Square 大法究竟是啥吧。
我们先来看看怎么用 Square 大法实现我在之前的博文中介绍的 SessionCalendarService 吧,详细代码例如以下:
/**
* Background {@link android.app.Service} that adds or removes session Calendar events through
* the {@link CalendarContract} API available in Android 4.0 or above.
*/
public class SessionCalendarService extends IntentService {
private static final String TAG = makeLogTag(SessionCalendarService.class);
//...
public SessionCalendarService() {
super(TAG);
}
@Override
protected void onHandleIntent(Intent intent) {
final String action = intent.getAction();
Log.d(TAG, "Received intent: " + action);
final ContentResolver resolver = getContentResolver();
boolean isAddEvent = false;
if (ACTION_ADD_SESSION_CALENDAR.equals(action)) {
isAddEvent = true;
} else if (ACTION_REMOVE_SESSION_CALENDAR.equals(action)) {
isAddEvent = false;
} else if (ACTION_UPDATE_ALL_SESSIONS_CALENDAR.equals(action) &&
PrefUtils.shouldSyncCalendar(this)) {
try {
getContentResolver().applyBatch(CalendarContract.AUTHORITY,
processAllSessionsCalendar(resolver, getCalendarId(intent)));
sendBroadcast(new Intent(
SessionCalendarService.ACTION_UPDATE_ALL_SESSIONS_CALENDAR_COMPLETED));
} catch (RemoteException e) {
LOGE(TAG, "Error adding all sessions to Google Calendar", e);
} catch (OperationApplicationException e) {
LOGE(TAG, "Error adding all sessions to Google Calendar", e);
}
} else if (ACTION_CLEAR_ALL_SESSIONS_CALENDAR.equals(action)) {
try {
getContentResolver().applyBatch(CalendarContract.AUTHORITY,
processClearAllSessions(resolver, getCalendarId(intent)));
} catch (RemoteException e) {
LOGE(TAG, "Error clearing all sessions from Google Calendar", e);
} catch (OperationApplicationException e) {
LOGE(TAG, "Error clearing all sessions from Google Calendar", e);
}
} else {
return;
}
final Uri uri = intent.getData();
final Bundle extras = intent.getExtras();
if (uri == null || extras == null || !PrefUtils.shouldSyncCalendar(this)) {
return;
}
try {
resolver.applyBatch(CalendarContract.AUTHORITY,
processSessionCalendar(resolver, getCalendarId(intent), isAddEvent, uri,
extras.getLong(EXTRA_SESSION_START),
extras.getLong(EXTRA_SESSION_END),
extras.getString(EXTRA_SESSION_TITLE),
extras.getString(EXTRA_SESSION_ROOM)));
} catch (RemoteException e) {
LOGE(TAG, "Error adding session to Google Calendar", e);
} catch (OperationApplicationException e) {
LOGE(TAG, "Error adding session to Google Calendar", e);
}
}
//...
}
如你所见,SessionCalendarService 调用了将要在后面定义的 helper 方法。一旦我们将这些 helper 方法和类的字段声明也考虑进来。Service 类的代码就有400多行。
要 hold 住这么庞大的类内发生的业务逻辑可不是什么简单的活,并且就像我们在上一篇博文中看到的那样。要在 SessionCalendarService 中进行单元測试简直是天方夜谭。
那如今来看看用 Square 大法实现它代码会是怎样的。我再强调一次:Square 大法须要我们将 Android 类内的业务逻辑迁移到一个业务对象中。
在这里,SessionCalendarService 所相应的业务对象则是 SessionCalendarUpdater。详细代码例如以下:
public class SessionCalendarUpdater {
//...
private SessionCalendarDatabase mSessionCalendarDatabase;
private SessionCalendarUserPreferences mSessionCalendarUserPreferences;
public SessionCalendarUpdater(SessionCalendarDatabase sessionCalendarDatabase,
SessionCalendarUserPreferences sessionCalendarUserPreferences) {
mSessionCalendarDatabase = sessionCalendarDatabase;
mSessionCalendarUserPreferences = sessionCalendarUserPreferences;
}
public void updateCalendar(CalendarUpdateRequest calendarUpdateRequest) {
boolean isAddEvent = false;
String action = calendarUpdateRequest.getAction();
long calendarId = calendarUpdateRequest.getCalendarId();
if (ACTION_ADD_SESSION_CALENDAR.equals(action)) {
isAddEvent = true;
} else if (ACTION_REMOVE_SESSION_CALENDAR.equals(action)) {
isAddEvent = false;
} else if (ACTION_UPDATE_ALL_SESSIONS_CALENDAR.equals(action)
&& mSessionCalendarUserPreferences.shouldSyncCalendar()) {
try {
mSessionCalendarDatabase.updateAllSessions(calendarId);
} catch (RemoteException | OperationApplicationException e) {
LOGE(TAG, "Error adding all sessions to Google Calendar", e);
}
} else if (ACTION_CLEAR_ALL_SESSIONS_CALENDAR.equals(action)) {
try {
mSessionCalendarDatabase.clearAllSessions(calendarId);
} catch (RemoteException | OperationApplicationException e) {
LOGE(TAG, "Error clearing all sessions from Google Calendar", e);
}
} else {
return;
}
if (!shouldUpdateCalendarSession(calendarUpdateRequest, mSessionCalendarUserPreferences)) {
return;
}
try {
CalendarSession calendarSessionToUpdate = calendarUpdateRequest.getCalendarSessionToUpdate();
if (isAddEvent) {
mSessionCalendarDatabase.addCalendarSession(calendarId, calendarSessionToUpdate);
} else {
mSessionCalendarDatabase.removeCalendarSession(calendarId, calendarSessionToUpdate);
}
} catch (RemoteException | OperationApplicationException e) {
LOGE(TAG, "Error adding session to Google Calendar", e);
}
}
private boolean shouldUpdateCalendarSession(CalendarUpdateRequest calendarUpdateRequest,
SessionCalendarUserPreferences sessionCalendarUserPreferences) {
return calendarUpdateRequest.getCalendarSessionToUpdate() == null || !sessionCalendarUserPreferences.shouldSyncCalendar();
}
}
我想要强调当中的一些要点:首先,须要注意。我们全然不须要用到不论什么新的关键字,由于业务对象的依赖都被注入了,它根本不会使用新的关键字,而这正是让类可单元測试的关键。其次。你会注意到类没有确切地依赖于 Android SDK,由于业务对象的依赖都是 Android 无关接口的 Android 特定实现。因此它不须要依赖于 Android SDK。
那么这些依赖是怎么加入到 SessionCalendarUpdater 类中的呢?是通过 SessionCalendarService 类注入进去的:
/**
* Background {@link android.app.Service} that adds or removes session Calendar events through
* the {@link CalendarContract} API available in Android 4.0 or above.
*/
public class SessionCalendarService extends IntentService {
private static final String TAG = makeLogTag(SessionCalendarService.class);
public SessionCalendarService() {
super(TAG);
}
@Override
protected void onHandleIntent(Intent intent) {
final String action = intent.getAction();
Log.d(TAG, "Received intent: " + action);
final ContentResolver resolver = getContentResolver();
Broadcaster broadcaster = new AndroidBroadcaster(this);
SessionCalendarDatabase sessionCalendarDatabase = new AndroidSessionCalendarDatabase(resolver,
broadcaster);
SharedPreferences defaultSharedPreferences = PreferenceManager.getDefaultSharedPreferences(this);
SessionCalendarUserPreferences sessionCalendarUserPreferences = new AndroidSessionCalendarUserPreferences(defaultSharedPreferences);
SessionCalendarUpdater sessionCalendarUpdater
= new SessionCalendarUpdater(sessionCalendarDatabase,
sessionCalendarUserPreferences);
AccountNameRepository accountNameRepository = new AndroidAccountNameRepository(intent, this);
String accountName = accountNameRepository.getAccountName();
long calendarId = sessionCalendarDatabase.getCalendarId(accountName);
CalendarSession calendarSessionToUpdate = CalendarSession.fromIntent(intent);
CalendarUpdateRequest calendarUpdateRequest = new CalendarUpdateRequest(action, calendarId, calendarSessionToUpdate);
sessionCalendarUpdater.updateCalendar(calendarUpdateRequest);
}
}
值得注意的是,改动后的 SessionCalendarService 到处都是新的关键字。但这些关键字在类中并不会引起什么问题。假设我们花几秒时间略读一下要点就会明确这一点:SessionCalendarService 类中已经没有不论什么业务逻辑,因此 SessionCalendarService 类不再须要进行单元測试。仅仅要我们确定在 SessionCalendarService 调用的是 SessionCalendarUpdater 类中的 updateCalendar() 方法,在 SessionCalendarService 唯一可能出现的就是编译时错误。
我们全然不须要为此实现測试单元。由于这是编译器的工作。与我们无关。
由于我在前两篇博文中提到的相关原因,将我们的 Service 类拆分成这样会使对业务逻辑进行单元測试变得非常easy,比如我们对 SessionCalendarUpdater 类进行单元測试的代码能够写成以下的样子:
public class SessionCalendarUpdaterTests extends TestCase {
public void testShouldClearAllSessions() throws RemoteException, OperationApplicationException {
SessionCalendarDatabase sessionCalendarDatabase = mock(SessionCalendarDatabase.class);
SessionCalendarUserPreferences sessionCalendarUserPreferences = mock(SessionCalendarUserPreferences.class);
SessionCalendarUpdater sessionCalendarUpdater = new SessionCalendarUpdater(sessionCalendarDatabase,
sessionCalendarUserPreferences);
CalendarUpdateRequest calendarUpdateRequest = new CalendarUpdateRequest(SessionCalendarUpdater.ACTION_CLEAR_ALL_SESSIONS_CALENDAR,
0,
null);
sessionCalendarUpdater.updateCalendar(calendarUpdateRequest);
verify(sessionCalendarDatabase).clearAllSessions(0);
}
}
结论
为了能够进行单元測试。我觉得改动后的代码变得更易读和更易维护了。能够肯定的是,我们还有很多办法能让代码变得更好。但在让代码能够进行单元測试的过程中,我想让改动后的代码尽可能与改动前风格类似,所以我没有进行其它改动。在下一篇博文中,我将会教大家怎样使用 Square 大法重构应用的 UI 组件(比如:Fragment 和 Activity)。
Android 进行单元測试难在哪-part3的更多相关文章
- android中单元測试中的断言assert的使用与扩展
首先看一组对照,比方说我们要測试的结果是一个Linearlaout AssertJ Android: assertThat(layout).isVisible() .isVertical() .has ...
- Android单元測试之JUnit
随着近期几年測试方面的工作慢慢火热起来.常常看见有招聘測试project师的招聘信息.在Java中有单元測试这么一个JUnit 方式,Android眼下主要编写的语言是Java,所以在Android开 ...
- (4.5.4)Android測试TestCase单元(Unit test)測试和instrumentationCase单元測试
Android单元和instrumentation单元測试 Developing Android unit and instrumentation tests Android的单元測试是基于JUnit ...
- Android下的单元測试
android下的单元測试 在AndroidManifest.xml文件里配置下面信息: 在manifest节点下加入: <!-- 指定測试信息和要測试的包 --> <instrum ...
- 【Android进阶】Junit单元測试环境搭建以及简单有用
单元測试的目的 首先.Junit单元測试要实现的功能,就是用来測试写好的方法是否可以正确的运行,一般多用于对业务方法的測试. 单元測试的环境配置 1.在AndroidManifest清单文件的Appl ...
- Android studio及eclipse中的junit单元測试
转载请标明出处:http://blog.csdn.net/nmyangmo/article/details/51179106 前一段时间有人问我单元測试的相关内容,我稍作总结做日志例如以下: 由于我接 ...
- 聊聊单元測试(一)——EasyMock
一.单元測试是保证软件质量的重要方法. 单元測试是对系统中某个模块功能的验证,但我们总会遇到这样那样的问题,导致測试代码非常难编写.最直接的一个原因便是强耦合关系,被測试者依赖一些不easy构造,比較 ...
- 利用Continuous Testing实现Eclipse环境自己主动单元測试
当你Eclipse环境中改动项目中的某个方法时,你可能因为各种原因没有执行单元測试,结果代码提交,悲剧就可能随之而来. 所幸infinitest(http://infinitest.github.io ...
- 在Eclipse中使用JUnit4进行单元測试(0基础篇)
本文绝大部分内容引自这篇文章: http://www.devx.com/Java/Article/31983/0/page/1 我们在编写大型程序的时候,须要写成千上万个方法或函数,这些函数的功能可能 ...
随机推荐
- codeforces 487E Tourists
如果不是uoj上有的话(听说这是China Round),我有可能就错过这道题目了(这是我有史以来为oi写的最长的代码,用了我一天TAT!). 题目 传送门. 一个连通无向图,点上有权,支持两种操作: ...
- linux下执行sh文件报错:oswatcher_restart.sh: line 13: ./startOSW.sh: Permission denied
1 查看执行sh文件的内容 [root@xxxdb0402 dbscripts]# more oswatcher_restart.sh #!/usr/bin/ksh #export oswdir=` ...
- 遭遇“HTTP 错误 500.19 无法访问请求的页面,因为该页的相关配置数据无效。”
windows 2008下IIS7 安装ASP.NET 遇到如下错误: HTTP 错误 500.19 - Internal Server Error 无法访问请求的页面,因为该页的相关配置数据无效. ...
- 终于懂了:Delphi消息的Result完全是生造出来的,不是Windows消息自带的(Delphi对Windows编程体系的改造越大,学习收获就越大)——消息是否继续传递就看这个Result
Windows中,消息使用统一的结构体(MSG)来存放信息,其中message表明消息的具体的类型, 而wParam,lParam是其最灵活的两个变量,为不同的消息类型时,存放数据的含义也不一样. t ...
- SOLR搭建企业搜索平台
一. SOLR搭建企业搜索平台 运行环境: 运行容器:Tomcat6.0.20 Solr版本:apache-solr-1.4.0 分词器:mmseg4j-1.6.2 词库:sogou-dic ...
- Java获取随机数的几种方法
Java获取随机数的几种方法 .使用org.apache.commons.lang.RandomStringUtils.randomAlphanumeric()取数字字母随机10位; //取得一个3位 ...
- bottle-session 0.2 : Python Package Index
bottle-session 0.2 : Python Package Index bottle-session 0.2 Download bottle-session-0.2.tar.gz Redi ...
- MYSQL获取自增主键【4种方法】
通常我们在应用中对mysql执行了insert操作后,需要获取插入记录的自增主键.本文将介绍java环境下的4种方法获取insert后的记录主键auto_increment的值: 通过JDBC2.0提 ...
- POJ - 1185 炮兵阵地 (状态压缩)
题目大意:中文题目就不多说大意了 解题思路: 1.每行最多仅仅有十个位置,且不是山地就是平原,那么就能够用1表示山地,0表示平原,将每一行的状态进行压缩了 2.接着找出每行能放炮兵的状态.先不考虑其它 ...
- android https通过载入pfx证书获取数据
直接给代码吧.研究了几天才搞定...... public static final String CLIENT_KET_PASSWORD = "Ku6OpqKDfN4=305790" ...