Android 进行单元测试难在哪-part2

发表于 5年以前  | 总阅读数:1799 次

Android 进行单元测试难在哪-part2

在上一篇博文中,我用干货告诉大家:即使是 Google 大牛写出来的代码也无法进行测试。确切地说,我真正告诉大家的是:根本没办法在 SessionDetailActivity 的 onStop() 方法里进行单元测试,而且详细地解释了个中因果:由于无法改变预测试状态,我们无法在 onStop() 方法里完成断言;在 onStop() 方法中进行测试时,获得测试后状态也是无法完成的。在上篇博文的结尾处,我跟大家说:正是 Android SDK 的某些特性,以及 Google 官方推荐的代码模板使得单元测试处于如此尴尬的境地,而且我承诺会在这篇博文中详尽地解释各种因由,那现在就让我来兑现我的诺言吧。

在我开始论述之前,我再说一次:正是标准的 Android 应用架构使测试 Android 应用变得如此困难,这句话是本系列博文的核心论点。这篇博文的意义在于:我们尝试提出理由证明重构 Android 应用的必要性,使得这些 Android 应用不需要明确地依赖于 Android SDK,与此同时,我们也尝试着提出一种健壮的应用架构,以增强 Android 应用的测试性,你会在这篇博文里了解到相关的概述。因此,我接下来将尝试去证明这篇博文的核心论点。

众所周知,开发 Android 应用有一种标准的架构,在示例代码和开源代码里很常见到应用的业务逻辑被放在 Android 应用的组件类,Activity,Service,Fragment 里执行。而我接下来就要遵循这种架构进行开发。而这篇博文要论述的就是:如果我们遵循这种标准架构进行开发,极有可能写下无法测试的代码,我在上一篇博文里也论证了这样的问题并不是偶然,正是标准的 Android 应用架构让测试变得支离破碎,单元测试几乎不能进行。

传统的 Android 应用架构让单元测试变得不可能

为了开始论证为什么标准开发架构让应用组件变得无法测试,大家不妨和我一起简要地复习下上篇博文的一些结论。单元测试包含三个步骤:准备,测试,断言。为了完成准备步骤,需要改变测试代码的预测试状态,此外,为了完成单元测试的断言步骤,我们需要获得程序的测试后状态。

复习了这些知识点后,可以开始进入正题了哈。在某些情况下,依赖注入是实现能够改变预测试状态代码的唯一办法,而且这些代码的测试后状态也是可访问的。我写了一个与 Android 完全无关的示例:

    public class MathNerd {

        private final mCalcCache;

        private final mCalculator;

        public MathNerd(CalculationCache calcCache, Calculator calculator) {
            mCalcCache = calcCache;
            mCalculator = calculator;
        }


        public void doIntenseCalculation(Calculation calculation, IntenseCalculationCompletedListener listener) {

            if (!mCalcCache.contains(calculation)) {

                mCalculator.doIntenseCalculationInBackground(listener);

            } else {

                Answer answer = mCalcCache.getAnswerFor(calculation);
                listener.onCalculationCompleted(answer);
            }
        }
    }

如上所示,依赖注入确实是对 doIntenseCalculation() 进行单元测试的唯一办法,因为 doIntenseCalculation() 方法根本没有返回值。除此以外,MathNerd 类里也没有判断测试后状态有效性的属性。但通过依赖注入,我们可以通过 mCalcCache 获得单元测试中的测试后状态。

    public void testCacheUpdate() {

        //Arrange
        CalculationCache calcCache = new CalculationCache();

        Calculator calculator = new Calculator();

        MathNerd mathNerd = new MathNerd(calcCache, calculator);

        Calculation calcualation = new Calculation("e^2000");

        //Act
        mathNerd.doIntenseCalculationInBackground(calculation, null);

        //some smelly Thread.sleep() code...

        //Assert
        calcCache.contains(calculation);
    }

如果我们这样做,很遗憾,恐怕是没办法为 MathNerd 类实现一个测试单元了。我们将会实现一个整合测试,用于检查 MathNerd 实际行为以及类是否根据 doIntenseCalculationInBackground() 方法处理后的值更新 CalcCache。

此外,依赖注入实际上也是验证测试单元测试后状态的唯一办法。我们通过注入验证方法在正确的位置被调用:

    public void testCacheUpdate() {

       //Arrange
        CalculationCache calcCache = mock(CalculationCache.class);

        when(calcCache.contains()).thenReturn(false);

        Calculator calculator = mock(Calculator.class);

        MathNerd mathNerd = new MathNerd(calcCache, calculator);

        Calculation calculation = new Calculation("e^2000");

        //Act
        mathNerd.doIntenseCalculationInBackground(calculation, null);

        //Assert should use calculator to perform calcluation because cache was empty
        verify(calculator).doIntenseCalculationInBackground(any());
    }

在 Android 应用的相关类中进行单元测试涉及的许多测试实例都需要一个东西:依赖注入。但问题来了:核心 Android 类持有我们无法注入的依赖。例如我上次提到的通过 SessionDetailActivity 启动的 SessionCalendarService 就是一个很好的例子:

    @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;
        }

       //...
    }

SessionCalendarService 的依赖是 ContentResolver,而且 ContentResolver 就是一个无法注入的依赖,所以如果我们并没有办法在 onHandleIntent() 方法里进行注入。而 onHandleIntent() 方法没有返回值,SessionCalendarService 类里也没有能让我们检查测试后状态的可访问的属性。为了验证测试后状态,我们可以通过查询 ContentProvider 检查请求数据是否被插入,但我们不会这样的方式为 SessionCalendarService 实现测试单元。相反,我们用的方法是实现一个整合测试,同时测试 SessionCalendarService 以及受 ContentProvider 操控的日历会议数据。

所以如果你把业务逻辑放在 Android 类里,而这个类的依赖又无法被注入,那这部分代码铁定没办法进行单元测试了。类似的无法被注入的依赖还有呢,例如:Activity 和 Fragment 的 FragmentManager。因此,至今为止 Google 官方一直鼓励我们使用的标准 Android 应用架构模式,教导我们在开发应用的时候要把业务逻辑放在应用的组件类里,信誓旦旦地说这是为我们好,而我们今天才知道真相竟然是:正是这样的架构让我们写下无法测试的代码。

标准开发模式让单元测试变得困难重重

某些情况下,标准的开发模式使代码的单元测试变得十分困难。如果我们回到上一篇博文提到的 SessionDetailActivity 里的 onStop() 方法,可以看到:

    @Override
    public void onStop() {
        super.onStop();
        if (mInitStarred != mStarred) {
            if (UIUtils.getCurrentTime(this) < mSessionStart) {
                // Update Calendar event through the Calendar API on Android 4.0 or new versions.
                Intent intent = null;
                if (mStarred) {
                    // Set up intent to add session to Calendar, if it doesn't exist already.
                    intent = new Intent(SessionCalendarService.ACTION_ADD_SESSION_CALENDAR,
                            mSessionUri);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_START,
                            mSessionStart);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_END,
                            mSessionEnd);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_ROOM, mRoomName);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_TITLE, mTitleString);
                } else {
                    // Set up intent to remove session from Calendar, if exists.
                    intent = new Intent(SessionCalendarService.ACTION_REMOVE_SESSION_CALENDAR,
                            mSessionUri);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_START,
                            mSessionStart);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_END,
                            mSessionEnd);
                    intent.putExtra(SessionCalendarService.EXTRA_SESSION_TITLE, mTitleString);
                }
                intent.setClass(this, SessionCalendarService.class);
                startService(intent);

                if (mStarred) {
                    setupNotification();
                }
            }
        }
    }

就像你看到的那样,onStop() 方法里压根没有能让我们知道 SessionCalendarService 是否通过正确的参数启动的可访问属性,此外,onStop() 方法是一个受保护的方法,使其返回值是无法修改的。因此,我们访问测试后状态的唯一办法就是检查注入到 onStop() 方法内的注入的状态。

这样一来,我们就会注意到 onStop() 方法中用于启动 SessionCalendarService 的代码并不属于某一个类。换句话说,onStop() 方法中注入的依赖根本不存在用于检查 SessionCalendarService 是否在正确的情况下通过正确的参数启动的测试单元测试后状态的属性。为了提出能让 onStop() 方法变为可测试的的第三种办法,那我们需要一些这样的东西:

    @Override
    public void onStop() {
        super.onStop();
        if (mInitStarred != mStarred) {
            if (UIUtils.getCurrentTime(this) < mSessionStart) {
                // Update Calendar event through the Calendar API on Android 4.0 or new versions.
                Intent intent = null;
                if (mStarred) {

                    // Service launcher sets up intent to add session to Calendar
                    mServiceLauncher.launchSessionCalendarService(SessionCalendarService.ACTION_ADD_SESSION_CALENDAR, mSessionUri, 
                                                                mSessionStart, mSessionEnd, mRoomName, mTitleString);

                } else {

                    // Set up intent to remove session from Calendar, if exists.
                    mServiceLauncher.launchSessionCalendarService(SessionCalendarService.ACTION_REMOVE_SESSION_CALENDAR, mSessionUri,
                                                                mSessionStart, mSessionEnd, mTitleString);
                }

                if (mStarred) {
                    setupNotification();
                }
            }
        }
    }

虽然这不是重构 onStop() 方法最简洁的方式,但如果我们按照标准开发方法把业务逻辑写在 Activity 里,并让写下的代码可以进行单元测试,类似的处理就变得必要了。现在不妨想想这种重构方式有多么违反常理:我们没有简单地调用 startService() 方法(startService() 是 Context 的一个方法,我们甚至可以说调用的是 SessionDetailActivity 的方法),而是通过依赖于 Context 的 ServiceLauncher 对象去启动该服务。SesionDetailActivity 作为 Context 的子类也将使用一个持有 Context 的对象去启动 SessionCalendarService。

不幸的是,即使我们像上面说的那样重构了 onStop() 方法,我们仍然不能保证能为 onStop() 方法实现测试单元。问题在于:ServiceLauncher 没有被注入,使得我们不能对 ServiceLauncher 进行注入,使我们能验证在测试过程中调用了正确的方法。

要对 ServiceLauncher进行注入,除了刚刚提到的以外,还会因为 ServiceLauncher 自身依赖于 Context 变得复杂,因为 Context 是一个非打包对象。因此,你并不能简单地通过将其传入用于启动 SessionDetailActivity 的 Intent 注入 ServiceLauncher。所以为了注入 ServiceLauncher,你需要开动你的小脑筋,或者使用类似于 Dagger¹ 的注入库。现在你应该也会发现,为了让我们的代码可以进行单元测试,我们确实需要完成许多复杂、繁琐的工作,而且,正如我即将在下篇博文中的论述,就算我们为了进行依赖注入而使用 Dagger 这样的库,在 Activity 内进行单元测试仍然是令人备受煎熬的。

为了让 onStop() 方法能进行单元测试,标准开发方式强迫我们使用反常理的重构方法,并要求我们在“根据以 Intent 为基础的依赖注入机制想出更好的重构方法”或“使用第三方的依赖注入库”。而标准开发方式为写下可测试代码带来的困难,就像在鼓励我们写下无法进行测试的代码,正是这种困难让我认为:标准开发方式阻碍我们写下可测试代码。

结论

在整个系列博文中,我一直在提出这样的观点:通过反思为什么在 Android 中进行单元测试如此困难,将帮助我们发现重构应用架构的各种好处,使我们的应用不必明确地依赖于 Android SDK。这篇博文论述到这里,我相信大家有足够理由相信完全摆脱 Android SDK 或许是个好提议了。

我刚刚把业务逻辑放在应用的组件类中,并向大家证明了对其进行单元测试有多么困难,甚至我们可以说对其进行单元测试这是不可能的。在下一篇博文中,我将建议大家将业务逻辑委托给使用了正确的依赖注入姿势的类。如果我们觉得定义这些类很麻烦的话,退而求其次,也能让这些类的依赖成为与 Android 无关的接口。与增强程序测试性的第一步相比,这一步是至关重要的,而完成第二步使我们无需 Android 特有的测试工具(例如:Roboletric,Instrumented Tests)就能写下更高效的测试单元。

  1. 毫无疑问,你在传入 ServiceLauncher 时应该使他变为一个序列化对象。但这并不是一个特别健壮的解决办法,因为只有在你不在乎序列化带来的性能影响时才能使用这个办法。
 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:8月以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:8月以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:8月以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:8月以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:8月以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:8月以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:8月以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:8月以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:8月以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:8月以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:8月以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:8月以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:8月以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:8月以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:8月以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:8月以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:8月以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:8月以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:8月以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:8月以前  |  398次阅读  |  详细内容 »
 相关文章
简化Android的UI开发 4年以前  |  520741次阅读
Android 深色模式适配原理分析 3年以前  |  28699次阅读
Android阴影实现的几种方案 1年以前  |  10869次阅读
Android 样式系统 | 主题背景覆盖 3年以前  |  9642次阅读
 目录