Android 进行单元測试难在哪-part3

Android 进行单元測试难在哪-part3

大家好,又见面了,我是全栈君。

在 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)。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/116358.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • 自带win10系统换win7的那些坑

    自带win10系统换win7的那些坑自带win10系统换win7的那些坑 这两天真是经历了一个换系统的巨坑。如果说这次换系统是一部历史的话那也一定是一部血泪史。今日4000多字的记录会把这部血泪史中的血和泪一一道出。不为别的只为:前车之鉴后事之师! 首先我对win10系统本身就很不喜欢,因为它不稳定兼容性差,还经常更新,用户总是给微软当小白鼠进行测试系统。之前就一直吐槽win10的自动更新无法…

  • vs的svn插件(怎么下载vs2010安装包)

    一.SVN客户端安装首先下载SVN地址:https://tortoisesvn.net/downloads.zh.html根据版本进行选择安装直接下一步下一步默认文件夹就好了安装完成右键就会有二个多出来的菜单都是英文不是很方便二.语言转换在下载SVN的下面就有语言包可以一起下载进行语言包安装找到c盘在这个目录下找到svn选择语言文件夹把语言包复制进去进行安装,安装完成打开SVN的settings的快捷方式,在里面进行选择语言,应用确认就好。这样一个SVN的安装汉化就完成

  • mysql中的varchar_从数据类型varchar转化为

    mysql中的varchar_从数据类型varchar转化为MySQL数据类型varchar详解更新时间:2014年03月17日11:10:11作者:这篇文章详细介绍了MySQL数据类型varchar,探讨varchar到底能存多长的数据、InnoDB和MyISAM中的varchar等问题,需要的朋友可以参考下1、varchar(N)的逻辑意义从MySQL4.1开始,varchar(N)中的N指的是该字段最多能存储多少个字符(characters…

  • php 工厂模式

    php 工厂模式一、 什么是工厂模式工厂类调用自身静态方法来生产对象实例工厂类:负责生成其他对象的类或方法途径:调用自身静态方法结果:一个实例对象工厂模式有一个关键的构造,根据一般原则命名为Fac

  • 移植 libuv 至 Visual C++ 6.0 并支持 Windows XP 编译系统

    移植 libuv 至 Visual C++ 6.0 并支持 Windows XP 编译系统

  • SQL Server 2012 数据库备份还原「建议收藏」

    SQL Server 2012 数据库备份还原「建议收藏」文章目录1.数据库备份2.创建备份设备使用SSMS工具创建备份设备使用SQL方式创建备份设备3.完整备份与还原使用SSMS工具完整备份与还原使用SQL方式完整备份与还原4.差异备份与还原使用SSMS工具差异备份与还原使用SQL方式差异备份与还原5.事务日志备份与还原使用SSMS工具事务日志备份与还原使用SQL方式事务日志备份与还原1.数据库备份    数据库备份,即从SQLServer数据…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号