• 原文链接 : 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);}@Overrideprotected 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);}@Overrideprotected 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相关推荐

  1. 太白---落燕纷飞第一重 Android单元測试Instrumentation和irobotium

    PS:叫太白---落燕纷飞纯粹好玩(天涯明月游戏画面感,打击感,碰撞尽管做的不尽人意,可是太白这个职业还是不错,用作开头,,做个旁白而已). 这里的单元測试不管是instrumentation还是ir ...

  2. Android单元測试之JUnit

    随着近期几年測试方面的工作慢慢火热起来.常常看见有招聘測试project师的招聘信息.在Java中有单元測试这么一个JUnit 方式,Android眼下主要编写的语言是Java,所以在Android开 ...

  3. (4.5.4)Android測试TestCase单元(Unit test)測试和instrumentationCase单元測试

    Android单元和instrumentation单元測试 Developing Android unit and instrumentation tests Android的单元測试是基于JUnit ...

  4. 【Android进阶】Junit单元測试环境搭建以及简单有用

    单元測试的目的 首先.Junit单元測试要实现的功能,就是用来測试写好的方法是否可以正确的运行,一般多用于对业务方法的測试. 单元測试的环境配置 1.在AndroidManifest清单文件的Appl ...

  5. atitit.jndi的架构与原理以及资源配置and单元測试实践

    atitit.jndi的架构与原理以及资源配置and单元測试实践 1. jndi架构 1 2. jndi实现原理 3 3. jndi资源配置 3 3.1. resin  <database> ...

  6. 使用maven运行单元測试总结

    maven本身没有单元測试框架,可是maven的default生命周期的test阶段绑定了maven-surefire-plugin插件,该插件能够调用Junit3.Junit4.TestNG等Jav ...

  7. 在Eclipse中使用JUnit4进行单元測试(0基础篇)

    本文绝大部分内容引自这篇文章: http://www.devx.com/Java/Article/31983/0/page/1 我们在编写大型程序的时候,须要写成千上万个方法或函数,这些函数的功能可能 ...

  8. 利用Continuous Testing实现Eclipse环境自己主动单元測试

    当你Eclipse环境中改动项目中的某个方法时,你可能因为各种原因没有执行单元測试,结果代码提交,悲剧就可能随之而来. 所幸infinitest(http://infinitest.github.io ...

  9. 单元測试中 Right-BICEP 和 CORRECT

    My Blog:http://www.outflush.com/ 在单元測试中,有6个总结出的值得測试的方面,这6个方面统称为 Right-BICEP.通过这6个方面的指导.能够较全然的測试出代码中的 ...

最新文章

  1. 面试官问我:spring、springboot、springcloud的区别,我笑了
  2. R语言应用实战-OLS模型算法原理及应用示例
  3. 使用事件和消息队列实现分布式事务
  4. nginx-配置基于ip或域名的虚拟主机
  5. 我的处女作《设计模式之禅》——前言
  6. mysql 文件系统规划_Mysql的文件系统规划以及日志配置
  7. 更新一波,微信第三方开发平台授权流程
  8. java集合课程,I学霸官方免费课程三十三:Java集合框架之Map集合
  9. .vue的文件在vscode里面是白色?
  10. eclipse -- git 提示
  11. oracle中的aix,oracle在AIX下的自启动
  12. ant调用YUI Compressor
  13. 本周成长记录及跟踪(2019年-11月-第4周)
  14. java代码防查重工具_代码查重工具sim
  15. elementUI表格合并行
  16. 39期1组,第一个项目感受---------文字与回忆
  17. 安卓开发 应用下载代码
  18. 脑电信号处理学习笔记(三)——pymrmr
  19. android ppsspp 存档位置,ppsspp怎么用,ppsspp怎么用psp存档
  20. 发现一个可视化大屏操作神器FBI,你值得一试

热门文章

  1. java 局部变量垃圾回收_java局部变量对垃圾回收的影响
  2. 20200225:最小路径和(leetcode64)
  3. 20200210:(leetcode 623)在二叉树中增加一行
  4. 借助计算机软件进行文学写作,网络文学创作对编辑提出的新要求及建议
  5. c 数组上限_高级I/O复用技术:Epoll的使用及一个完整的C实例含代码
  6. oracle将日期格式化to_char及字符串转日期to_date
  7. js中src赋值理解
  8. MIPI - DVP
  9. 从RedHat到MongoDB,开源商业软件是如何占领世界的
  10. 上海交大发布全球首款专用光量子计算软件