iis服务器助手广告广告
返回顶部
首页 > 资讯 > 移动开发 >Android中的单元测试
  • 353
分享到

Android中的单元测试

单元单元测试测试Android 2022-06-06 12:06:38 353人浏览 八月长安
摘要

  随着Agile的普及,以及开发人员对测试重要性的认识逐步加深,单元测试已经成了越来越多软件项目开发中不可缺少的一部分。无论项目是不是采用TDD的形式来进行开发,单元测试

  随着Agile的普及,以及开发人员对测试重要性的认识逐步加深,单元测试已经成了越来越多软件项目开发中不可缺少的一部分。无论项目是不是采用TDD的形式来进行开发,单元测试都能够为项目的修改和重构提供一定的保障。   Android作为主要的移动平台之一,吸引了无数的开发人员。但面对Android平台和环境的种种限制,很多开发人员往往有心无力,很难为其项目添加全面有效的单元测试。   Android平台的开发环境中集成了一个测试框架(Instrumented Test),用于支持其单元测试和验收测试。Robotium同样提供一个类似于selenium的测试框架,使得开发人员可以对应用的功能进行验证。这两种方式提供的测试环境都类似于集成测试,它们的测试用例都需要运行在模拟器上,通过对模拟器的操作或者mock,来触发函数调用,进而对其结果进行验证。这种方法通常粒度较大,测试的编写和维护较为困难,而为重要的是,由于测试需要运行于模拟器或测试机器上,我们在运行前需要将测试和应用打包,进行部署安装,并终运行在模拟器或测试机器的Delvik虚拟机上,其运行速度较普通的单元测试要慢许多,如果使用TDD来进行开发,根本无法达到快速开发的要求。   之所以这些框架的测试用例都需要在模拟器中运行,是因为我们平时在开发时所使用的Andorid.jar是被精简过的,只是用于日常开发的,它只是一个placeholder,使得我们在开发时能够不出编译错误,它完全是一个stub包,其中所有的类都只是Android平台接口的一个stub,如果在代码中运行这个Android.jar,它们所有的方法都只会抛出一个java.lang.RuntimeException(“Stub!”)异常。所以,一旦测试代码需要真正调用Android平台相关的类或接口,它们必须运行于真正实现了Android的环境,如模拟器或者是测试机器。   我们的另外一个选择是只对POJO进行单元测试,如果遇到Android相关的代码,使用Mock框架对其进行模拟。这种方式一定程度上可以解决我们的问题,但这意味着我们需要大量的在测试环境中使用mock和stub。另外,虽然Android中界面的布局通常使用XML来实现,但项目的代码中还是会存在各种对界面的操作和更新,UI和逻辑的耦合使测试更加不易。   而且即使这样,由于Android平台的复杂性(static方法,final方法和类,Context和Resources的管理),我们也很难对Android相关的代码进行测试,以保证测试率。   那么如何能够在不增加开发成本的情况下,有一个稳定快速的单元测试环境呢?   我们目前的选择是使用MVP模式和Robolectric   Android的体系结构非常适合于使用MVP模式进行开发,与mvc模式不同,Android中的Activity并不是一个标准的Controller,它的首要职责是加载应用的布局和初始化用户界面,并接受并处理来自用户的操作请求,进而作出响应。随着界面及其逻辑的复杂度不断提升,Activity类的职责不断增加,以致变得庞大臃肿。当我们将其中复杂的逻辑处理移至另外的一个类(Presneter)中时,Activity其实是MVP模式中View,它负责UI元素的初始化,建立UI元素与Presenter的关联(Listener之类),同时自己也会处理一些简单的逻辑(复杂的逻辑交由Presenter处理)。如图所示:

  通过这种职责的分离,一方面代码的可读性得到了提高,另一方面我们可以更为方便地通过mock Activity的方式对各种逻辑(Presenter中的方法)进行测试。   对于测试环境的搭建和测试Android相关的代码,我们则借助于Robolectric的帮助。   Robolectric在其所提供的测试框架中,完全模拟了Android SDK的jar文件(不会再有恼人的stub异常),它使得我们的测试可以运行于JVM之上(速度得到大幅度的提升),因此我们可以用它对Android应用进行测试驱动开发。Roblectric同时实现了Android中对XML的解析,模拟了View,Layout,以及资源的加载,它使得Android的环境对于开发人员来说更像是一个黑盒,从而使开发人员不用大量使用mock,可以方便的对资源状态和Android相关的代码进行测试。

Robolectric是如何做到这点的呢?   Robolectric使用了javassist在运行时动态修改Android.jar中类的byte code,Robolectric会在JVM加载Android.jar包的时候,重写其中类的方法。Roblectroic会让这些方法有返回值(null或是0)而不是抛出异常,或者将这些方法调用转向Shadow Objects来模拟Android SDK的实现。Shadow Objects是Robolectric在运行时插入到Android.jar包相应的类中的,它们会实际处理方法的调用,并记录相应的状态,以备在assert的时候进行查询。如图所示。Robolectric提供了大量的Shadow Objects,覆盖了测试开发过程中绝大多数逻辑功能的需要。

  Robolectric的使用   基于Robolectric的测试需要使用其特定的test runner(RobolectricTestRunner)来运行,我们可以通过扩展RobolectricTestRunner来创建一个自己的test runner,并在其构造函数中设定需要加载的AndroidManifest.xml和resource目录。如:   public class MyTestRunner extends RobolectricTestRunner{   public MyTestRunner(Class<?>testClass)throws InitializationError{   super(testClass,new RobolectricConfig(new File("my_app/AndroidManifest.xml"),new File("my_app/res")));   }   }   有了自己的test runner,我们可以来写一个简单的Robolectric测试了   1@RunWith(MyTestRunner.class)   public class SignInScreenTest{   @Test   public void should_start_intent_when_click_reGIStration_button(){   2 Activity activity=new Activity();   SignInScreen signInScreen=new SignInSceen(activity);   3 TextView textView=(TextView)signInScreen.findViewById(R.id.sign_in_registration);   textView.perfORMClick();   4 ShadowActivity shadowActivity=Robolectric.shadowOf(activity);   Intent nextStartedActivity=shadowActivity.getNextStartedActivity();   ShadowIntent shadowIntent=Robolectric.shadowOf(nextStartedActivity);   assertThat((Class<WEBPageActivity>)shadowIntent.getIntentClass(),equalTo(WebPageActivity.class));   }   }   在这段测试代码中:   (1)声明了测试运行的test runner;像普通的单元测试,它也分为了set up,method invoke,以及assert三个阶段。   在(2)中,测试初始化了一个Activity用于提供Context,并使用这个Activity对象生成了一个SignInScreen实例;   第二个阶段,也是(3)中,代码在生成的登录界面中找到注册按钮,并进行点击。为有意思的第三个阶段需要验证注册按钮的点击触发了我们期望的事件,即使用Implicit Intent来打开WebPageActivity。   为了进行这个验证,(4)中首先通过Robolectric的静态方法shadowOf来获取activity对象相应的Shadow Object,而通过这个Shadow Object,代码获得了activity对象的所开启的Intent对象。后通过Intent对象的Shadow Object,我们可以获得其intent class并进行验证。   通过这个测试我们可以看到,有了Robolectric的帮助,我们可以轻松的生成Activity实例,加载xml布局文件,进行组件上的方法调用。通过shadow对象,我们则可以获取Android相关类的对象状态信息,来对测试的结果进行验证。实际上除了Intent,我们还可以对通过使用Robolectric对代码中的Dialog,Http请求,数据库操作等各个方面进行测试。   Robolectric并没有为Android SDK中的所有类都定义shadow对象,你可以通过调用Robolectric.getDefaultShadowClasses()方法来查看你所需要的类是否已经被注册到了需要被shadow的类列表中。如果没有你可能需要对其进行定制和扩展。关于如何添加Shadow Objects而增加Robolectric的功能,在Robolectric的网站文档中有详细的描述。   由于Robolectric的测试是可以脱离Android的SDK运行于JVM上,我们可以像运行普通的jUnit测试一样在IDE中或者在终端使用构建脚本运行我们的测试。   由于Robolectric的更新并不是很频繁,我们在平时也遇到了一些需要定制的情况,如支持Android4.0,使用sonar进行项目质量分析等等。所以我们在GitHub上fork了Robolectric的工程,并以git submodule的方式将其加入到我们的工程管理中来,这样,我们可以根据自己的需要来对Robolectric进行修改和扩展。由于我们对Robolectric的修改频率非常的低,在每一次修改后,可以将其编译打包成一个jar文件,将这个jar文件加入到我们的工程管理中,让我们的测试代码仍然依赖于这个jar文件,这样可以免去在运行测试中不必要的对Robolectric的重复编译,加快测试代码的运行速度。   我们在当前的项目中也进行了一定的关于验收测试方面的尝试,由于测试脚本是开发人员与BA以及QA进行沟通的一种重要途径,也是开发人员和QA进行人工测试的基准,因此我们仍然选用了cucumber作为我们编写脚本的工具,再使用cuke4duke和jRuby对其进行解析和执行。但目前这种测试方式似乎并不成熟,我们在这种尝试和实践的过程中遇到了种种的问题,主要在于测试编写和维护上的困难,这也导致了我们验收测试的覆盖率并不高。我们也会在这一方向上进行更多的尝试,如果大家有更好的关于验收测试自动化方面的实践,也希望能够得到你们的帮助和指正。


--结束END--

本文标题: Android中的单元测试

本文链接: https://www.lsjlt.com/news/28777.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • android单元测试怎么实现
    Android单元测试可以通过使用JUnit框架和Android Testing Support Library来实现。以下是实现A...
    99+
    2023-08-29
    android
  • android单元测试如何配置
    要配置Android单元测试,您可以按照以下步骤进行操作:1. 在您的Android项目中,打开`build.gradle`文件。2...
    99+
    2023-09-26
    android
  • python中的单元测试和数据库测试
    登录测试被测试的代码逻辑 @app.route('/login', methods=['POST']) def login(): username = request.form.get('username') passwor...
    99+
    2023-01-31
    单元测试 数据库 测试
  • java-单元测试
    一、什么是单元测试? 单元测试(unit testing),是指对软件中的最小可测试单元进行检查和验证。至于“单元”的大小或范围,并没有一个明确的标准,“单元”可以是一个函数、方法、类、功能模块或者子系统。 单元测试通常和白盒测试联系到...
    99+
    2023-09-26
    java 学习 Powered by 金山文档
  • Android Studio下的单元测试怎么编写
    在Android Studio中编写单元测试可以使用JUnit框架来进行测试。以下是编写Android Studio下单元测试的基本...
    99+
    2023-10-21
    Android
  • Python 单元测试 & 文档测试
    1.1   单元测试1.1.1   单元测试编写单元测试是用来对一个模块、一个函数或者一个类来进行正确性检验的测试工作。编写一个Dict类,这个类的行为和dict一致,但是通过属性来访问。>>> d = Dict(a=1,...
    99+
    2023-01-31
    单元测试 文档 测试
  • Python测试 | 在python中创建单元测试
    作为开发人员或程序员, 只要编写代码,通常就会忽略一件事:测试。测试是开发的一部分,您应该严格测试应用程序,以确保它们满足必需的功能。让我们先从为什么要做测试开始:1、减少手动测试的需求因为测试用例是预先编写的,所以没有真正的理由再进行手动...
    99+
    2023-05-14
    Python 测试 单元
  • PHP中的单元测试框架
    随着软件开发领域的迅速发展,软件测试的重要性也越来越被大家所重视。单元测试是软件测试中的一个重要环节,它能够在程序开发的早期就发现潜在的问题,从而提高软件的质量和稳定性。而在PHP语言领域中,有许多非常优秀的单元测试框架,本文将介绍其中的一...
    99+
    2023-05-23
    框架 PHP 单元测试
  • @SpringBootTest单元测试测试类的使用
    前言 使用SpringBoot 测试类可在不需要启动程序时,即可使用。当你运行你的测试方法时他会自己启动程序调用所需使用到的mapper,service接口,实现方法。故而可在测试类中像编写正常service方法一样编写代码 一.依赖录入 ...
    99+
    2023-08-21
    单元测试 junit java
  • 测试TLS客户端的单元测试
    在Golang实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天编程网就整理分享《测试TLS客户端的单元测试》,聊聊,希望可...
    99+
    2024-04-04
  • PHP中的单元测试和集成测试指南
    随着软件开发的不断发展,测试已经成为保证软件质量的必要步骤。而随着PHP语言的广泛应用,PHP开发中的单元测试和集成测试也愈发重要。在本篇文章中,我们将会探讨以下主题:什么是单元测试和集成测试单元测试和集成测试的优势PHPUnitPHPUn...
    99+
    2023-05-21
    PHP 单元测试 集成测试
  • 怎么在Android中利用Activity实现单元测试
    这期内容当中小编将会给大家带来有关怎么在Android中利用Activity实现单元测试,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。对Acitivity的测试对于Activity,我们大致有两种测试需求...
    99+
    2023-05-30
    android activity
  • 如何在 Golang 单元测试中调试失败的测试?
    非常抱歉,由于您没有提供文章标题,我无法为您生成一篇高质量的文章。请您提供文章标题,我将尽快为您生成一篇优质的文章。...
    99+
    2024-05-16
  • 【Spring Boot】单元测试
    单元测试 单元测试在日常项目开发中必不可少,Spring Boot提供了完善的单元测试框架和工具用于测试开发的应用。接下来介绍Spring Boot为单元测试提供了哪些支持,以及如何在Spring B...
    99+
    2023-09-16
    spring boot 单元测试 log4j
  • PHP开发中的单元测试与自动化测试
    php 中的单元测试和自动化测试至关重要,可确保代码可靠性。单元测试隔离测试代码单元,通过断言检查预期结果。自动化测试利用 ci 平台自动执行测试,确保代码持续满足质量标准,包括配置测试...
    99+
    2024-05-09
    自动化测试 单元测试 git
  • Python 单元测试(unittest
    项目的整体结构可以参考“软件目录开发规范”,这里单说测试目录。一般都是在项目里单独创建一个测试目录,目录名就是“tests”。关于目录的位置,一种建议是,在项目名(假设项目名是Foo)的一级子目录下创建二级子目录 “Foo/foo/tes...
    99+
    2023-01-31
    单元测试 Python unittest
  • Spring Boot 单元测试
    文章目录 1. 单元测试是什么2. 单元测试的优点3. 进行 Spring Boot 单元测试3.1 确认项目中已经内置了测试框架3.2 生成单元测试的类3.3 添加 @SpringBootT...
    99+
    2023-09-15
    单元测试 spring boot java
  • 浅谈Android单元测试的作用以及简单示例
    前提概要受人嫌弃的单元测试对于单元测试这个知识点,其实很多开发者是不太接触的,包括笔者,在实习之前也并未实用过单元测试,或者说并没感受到单元测试的好处。 对于bug的调试,笔者之前更倾向于使用log和断点调试,可以说会了这两个,大部分的逻辑...
    99+
    2023-05-30
    android 单元测试
  • 单元测试与集成测试在golang函数测试中的区别?
    单元测试和集成测试是两种不同的 go 函数测试类型,分别用于验证单个函数或多个函数的交互和集成。单元测试只测试特定函数的基本功能,集成测试测试多个函数之间的交互和与应用程序其他部分的集成...
    99+
    2024-04-27
    单元测试 集成测试 golang
  • golang函数的单元测试
    单元测试测试单个函数或小片段逻辑,以确保修改后代码仍按预期运行。实战案例包括编写函数、创建测试文件、定义测试用例并使用 t.errorf 报告测试失败。最佳实践包括为每个函数编写测试、使...
    99+
    2024-04-20
    golang 单元测试 git
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作