`
dowhathowtodo
  • 浏览: 805660 次
文章分类
社区版块
存档分类
最新评论

android基础知识12:android自动化测试06—Instrumentation 02 单元测试

 
阅读更多

【IT168 技术文档】任何程序的开发都离不开单元测试来保证其健壮和稳定。Android的程序自然也不例外。从Android SDK 0.9开始,就有了比较成熟的测试框架,但是直到目前最新的1.1版本,也没有详细的文档介绍这个内容,只是简单的给了一个Api Demos里的几个单元测试代码。因此,我在这里对此内容做一下梳理和总结:


  JUnit还能用么?

  在 Java下做单元测试必然用到JUnit。这里说的JUnit是指从Apache基金会下载的junit.jar里提供的一系列单元测试功能。这些功能显然是运行在JDK之上的。在Android下已经没有了JDK,自然也无法运行JUnit。但是这并不妨碍我们利用JUnit编写单元测试。只不过在运行单元测试时,一定要用JDK来运行,利用java命令来启动JUnit的某个Runner。如果是用Eclipse的话,可以在Run Configuration里新建一个JUnit。但是一定要记得在Classpath选项卡里将Bootstrap Entries中的Android Library改成JRE,并且添加junit.jar。


很明显的,这种测试就是正规的Java单元测试,和Android没有任何关系。你无法测试任何关于Android系统中的API,你写的Activity,人机界面等等。所以,如果你想测试仅仅是一些封装数据的对象,或者是纯粹的数值计算,还是可以用这种方法的。

  Android里面的junit.framework包是怎么回事?

  很多人看到这个包的时候,第一反应是Android是不是已经完整集成了JUnit。很遗憾这不是事实。如果你按照JUnit的运行方法,却不像上面那样改用JDK,就一定会得到一个异常:

  #

  # An unexpected error hasbeen detected by Java Runtime Environment:

  #

  # Internal Error(classFileParser.cpp:2924), pid=4900, tid=4476

  #Error:ShouldNotReachHere()

  #

  # Java VM: JavaHotSpot(TM) Client VM (10.0-b19 mixed mode windows-x86)

  # An error report filewith more information is saved as:

  #E:\Mydoc\EclipseWorkspace\TestAndroid\hs_err_pid4900.log

  #

  # If you would like tosubmit a bug report, please visit:

  #http://java.sun.com/webapps/bugreport/crash.jsp

  #

  实际上,TestCase这个类用于在Android担当所有独特的TestCase的基类的作用,它是一个Abstract ClassAndroid单元测试类继承关系图如下所示:

之所以有那么多XXXTestCase主要是为了简化工作。例如当你想对一个访问数据库的功能进行测试时,首先需要自己启动并初始化数据库。在这里是类似的,如果你想测试一个Activity,首先要启动它。而ActivityTestCase就会自动帮你做完这些事情。而 ActivityUnitTestCase会更注重测试的独立性,它会让测试与Android底层的联系降到最低。其余的类可以查看相关的Javadoc 来按需挑选。要编写测试,就是找到合适的XXXTestCase作为基类来继承,并且编写自己的测试方法。
  很明显的,最简单的编写测试的方法就是继承AndroidTestCase写一个自己的TestCase。然后为自己的一组TestCase写一个Activity界面,由界面控制 TestCase的启动,运行和结果报告。但是,你很快会发现,为何要给测试写一个界面呢?这太诡异了。这时就需要一种技术,它可以利用命令行(Shell)来启动一组测试,并且通过命令行的形式给出结果。这就是所谓的Instrumentation。


什么是Instrumentation?


  一般在开发Android程序的时候,需要写一个manifest文件,其结构是:

<application android:icon="@drawable/icon" android:label="@string/app_name">
<activity android:name=".TestApp" android:label="@string/app_name">
……
</activity>
</application> 

这样,在启动程序的时候就会先启动一个Application,然后在此Application运行过程中根据情况加载相应的Activity,而Activity是需要一个界面的。但是Instrumentation并不是这样的。你可以将Instrumentation理解为一种没有图形界面的,具有启动能力的,用于监控其他类(用Target Package声明)的工具类。任何想成为Instrumentation的类必须继承android.app.Instrumentation。下面是这个类的解释:

  “Base class for implementingapplication instrumentation code. When running with instrumentation turned on,this class will be instantiated for you before any of the application code,allowing you to monitor all of the interaction the system has with the application.An Instrumentation implementation is described to the system through anAndroidManifest.xml's tag.“

  对于单元测试,我们需要认真了解的就是android.test.InstrumentationTestRunner类。这是Android单元测试的主入口。它相当于JUnit当中TestRunner的作用。

  那么如何加载它呢,首先要在manifest文件中加入一行关于Instrumentation的声明。比如Android Api Demos中的测试里的manifest是这么写的(我滤掉了所有的注释):

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.apis.tests">
    <application>
       <uses-library android:name="android.test.runner" />

    </application>

    <instrumentation android:name="android.test.InstrumentationTestRunner"

      android:targetPackage="com.example.android.apis"

        android:label="Tests for Api Demos."/>

</manifest> 
如果用Eclipse的ADT插件(0.8版本以上),也可以用图形界面来添加,如下图:

  编辑好 manifest,就可以打包(build,可以用Eclipse ADT来做,也可以用aapt命令手工完成),然后安装到虚拟机上(用adb install命令)。之后就可以利用命令行的方式来加载你的单元测试了。在Android Shell中加载一个Instrumentation的方法是利用以下命令:

  adb shell am instrument –w XXXXXX

  其中-w是指定Instrumentation类的参数标志。一个简单的例子是:

  adb shell am instrument -wcom.android.foo/android.test.InstrumentationTestRunner

  当然,也可以利用adb shell先进入android命令行模式,再直接写am instrument –w XXXXXXX。下面将具体介绍如何将根据需要加载一组单元测试。

如何在Android中利用Instrumentation来进行测试?

  在介绍具体的命令之前,我们先理解一下单元测试的层次。一组单元测试可以被组织成若干个TestSuite。每个TestSuite包含若干 TestCase(某个继承android.jar的junit.framework.TestCase的类)。每个TestCase又包含若干个 Test(具体的test方法)。

  如果假设com.android.foo是你的测试代码的包的根。当执行以下命令时,会执行所有的TestCase的所有Test。测试的对象就是在Target Package中指定的包中的代码:

  adb shell am instrument -wcom.android.foo/android.test.InstrumentationTestRunner

  如果你想运行一个TestSuite,首先继承android.jar的junit.framework.TestSuite类,实现一个 TestSuite(比如叫com.android.foo.MyTestSuite),然后执行以下命令执行此TestSuite

  adb shell am instrument -e classcom.android.foo.MyTestSuite -wcom.android.foo/android.test.InstrumentationTestRunner

  其中的-e表示额外的参数,语法为-e [arg1] [value1] [arg2] [value2]…这里用到了class参数。

  如果仅仅想运行一个TestCase(比如叫com.android.foo.MyTestCase),则用以下命令:

  adb shell am instrument -e classcom.android.foo.MyTestCase -wcom.android.foo/android.test.InstrumentationTestRunner

  如果仅仅想运行一个Test(比如就是上面MyTestCase的testFoo方法),很类似的,就这样写:

  adb shell am instrument -e classcom.android.foo.MyTestCase#testFoo -wcom.android.foo/android.test.InstrumentationTestRunner

  然后,所有的测试结果会输出到控制台,并会做一系列统计,如标记为E的是Error,标记为F的是Failure,Success的测试则会标记为一个点。这和JUnit的语义一致。如果希望断点调试你的测试,只需要直接在代码上加上断点,然后将运行命令参数的-e后边附加上debug true后运行即可。更加详细的内容可以看InstrumentationTestRunner的Javadoc。我希望Android能尽快有正式的文档来介绍这个内容。

  如何在Android的单元测试中做标记?

  在 android.test.annotation包里定义了几个annotation,包括 @LargeTest,@MediumTest,@SmallTest,@Smoke,和@Suppress。你可以根据自己的需要用这些 annotation来对自己的测试分类。在执行单元测试命令时,可以在-e参数后设置“size large”/ “size medium”/ “size small”来执行具有相应标记的测试。特别的@Supperss可以取消被标记的Test的执行。

  完整的操作过程

  总结以上所有的内容,编写并运行完整的测试需要以下的步骤:

  以上步骤中,在 Android自带的例子中,我发现它有两个manifest.xml。也就是说在步骤3中源代码和测试代码分别生成了两个不同的包。然后步骤4利用 adb install命令安装到了虚拟机上。由于我没有找到Eclipse ADT有办法可以为一个只有Instrumentation,没有Activity的Application打包并安装,于是采用了略微不同的办法完成了这个工作。下文中将一一详细介绍整个过程。

1、编写程序

  我新建了一个项目TestApp,参数为:

  Package Name: com.android.testapp

  Activity Name: MainActivity

  Application Name: TestApp

  以下是MainActivity的源代码:

packagecom.android.testapp;
  importandroid.app.Activity;
  importandroid.os.Bundle;
  publicclassMainActivityextendsActivity {
  /** Called when the activity is first created. */
  @Override
  publicvoidonCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);
  }
  publicintsum(inta,intb) {
  returna + b;
  }
  publicintsubstract(inta,intb) {
  returnb - a;
  }
  }

其中,我故意将减法的a – b写成了b – a。

2、编写测试程序

  然后,我新建了一个Source Folder,名为test,并在里面新建了包com.android.testapp.test。并定义了一个TestCase,名为TestMainActivity,源代码如下:

package com.android.testapp.test;
  import com.android.testapp.MainActivity;
  import android.test.ActivityInstrumentationTestCase;
  import android.test.suitebuilder.annotation.MediumTest;
  public class TestMainActivity extends ActivityInstrumentationTestCase {
  public TestMainActivity() {
  super("com.android.testapp", MainActivity.class);
  }
  public TestMainActivity(String pkg, Class activityClass) {
  super(pkg, activityClass);
  }
  @MediumTest
  public void testSum() {
  assertEquals(3, getActivity().sum(1, 2));
  }
  @MediumTest
  public void testSubstract() {
  assertEquals(-1, getActivity().substract(1, 2));
  }
  }

我继承了ActivityInstrumentationTestCase。这个TestCase在执行时会自动帮我启动相应的Activity。

  接下来就是程序的Manifest:

<?xml version="1.0" encoding="utf-8"?> 

<manifest xmlns:android="http://schemas.android.com/apk/res/android" 

package="com.android.testapp" 

android:versionCode="1" 

android:versionName="1.0.0"> 

<application android:icon="@drawable/icon" android:label="@string/app_name"> 

<activity android:name=".MainActivity" 

android:label="@string/app_name"> 

<intent-filter> 

<action android:name="android.intent.action.MAIN" /> 

<category android:name="android.intent.category.LAUNCHER" /> 

</intent-filter> 

</activity> 

<uses-library android:name="android.test.runner" /> 

</application> 

<instrumentation android:targetPackage="com.android.testapp" android:name="android.test.InstrumentationTestRunner" android:label="Test Unit Tests"></instrumentation> 

</manifest> 

在这个文件中,我将 Activity和Instrumentation的声明写到了一起,而没有像Apis Demo那样分开。请注意里面的标签。如果没有那句,在运行测试时会报告找不到TestRunner。这是由于 Android在build的时候只把需要的东西打包,所以你必须明确的告诉Android Builder这一点。

3、Build和Install

  在 Eclipse上,这两个步骤是一起完成的。只要点一下Run即可。只不过如果你不在Run Configuration里将安装后的Launch Action设为“Do Nothing”,就会自动运行一下你的MainActivity。对于我们,设为Do Nothing即可。如下图:



完成后,利用命令:

  adb shell pm list packages

  可以在已经安装的pkg列表里看到com.android.testapp。

  4、运行测试,查看结果

  之后就打开命令行,运行以下命令

  adb shell am instrument –e classcom.android.testapp.test.TestMainActivity –wcom.android.testapp/android.test.InstrumentationTestRunner

  即可看到如下的结果:



  可以看到,单元测试正确的找到了减法中的错误。结果中的成功的测试显示为”.”,一个失败的显示为”F”。只不过我还是不太理解为什么我只写了两个测试方法,Tests run却显示了3。


参考资料:

android上的单元测试


分享到:
评论

相关推荐

    Android自动化测试之Robotium--基础操作

    根据提供的文件信息,接下来详细阐述Android自动化测试工具Robotium的基本知识点。 首先,Robotium是一个开源的自动化测试框架,专门用于Android平台的应用程序。它弥补了ActivityInstrumentationTestCase2在集成...

    Android自动化测试之Robotium--进阶操作.pdf

    Android自动化测试是移动应用开发过程中的一个重要环节,它能够帮助开发者和测试人员快速发现应用中可能存在的问题,并验证应用功能的正确性。在Android自动化测试的众多工具中,Robotium是目前广泛使用的一款自动化...

    Android单元测试源码.zip

    8. Instrumentation测试:与纯Java单元测试不同,Android的单元测试通常被称为Instrumentation测试,因为它们需要在Android环境中运行,并由一个称为Instrumentation的进程控制。 9. Robolectric:对于不涉及UI的...

    深入浅出Android 自动化测试1

    Android自动化测试是提升应用质量与开发效率的关键环节。本文将深入探讨Android自动化测试的不同方面,包括测试工具、分类、实践以及代码覆盖率。 首先,我们来看看常见的Android自动化测试工具。Monkey、...

    Appium自动化测试工具介绍

    ### Appium自动化测试工具知识点详解 #### 一、Appium简介及原理 ##### 1.1 Appium简介 - **定义**: Appium是一款强大的开源自动化测试工具,支持iOS和Android两大移动操作系统,能够进行跨平台的测试工作。适用...

    android自动化测试robotium之adb shell的使用

    在Android自动化测试中,ADB(Android Debug Bridge)是一个强大的工具,用于与Android设备进行通信,包括模拟器和...理解这些基础知识对于进行Android自动化测试至关重要,能够帮助开发者高效地调试和验证应用的功能。

    0android_test[1]

    下面我们将深入探讨Android测试的基础知识、Android Studio的测试工具以及"HelloWorld"测试案例。 1. Android测试基础: Android测试主要分为单元测试、功能测试、集成测试和系统测试。单元测试关注于单一代码模块...

    Appium mac/android自动化

    学习Appium自动化的过程中,需要具备Android开发的基础知识、熟悉Android SDK工具和知道如何使用命令行。同时,了解Java或Python语言基础对于编写和理解测试脚本非常重要。通过这些准备工作和学习内容,开发者可以...

    Android test测试小demo

    JUnit是Java领域广泛使用的单元测试框架,Android也支持其作为基础的测试工具。Espresso则是一个UI测试框架,它允许开发者编写直接操作UI元素并验证其行为的测试代码。通过Espresso,你可以确保用户界面的各个组件...

    Android移动应用测试实战部分课件及案例.zip

    3. **Android测试技术**:《android测试技术.ppt》涵盖了Android平台特有的测试工具和技术,如 Espresso(UI自动化测试框架)、Robolectric(模拟Android环境进行单元测试)、Instrumentation测试等,使开发者能够在...

    安卓UiAutomator/UiDevice/Instrumentation官方测试范例

    通常,Instrumentation用于编写单元测试和功能测试,可以配合JUnit等测试库使用。在`UiAutomator-BasicSample`中,虽然主要展示的是UiAutomator的用法,但理解Instrumentation对于完整测试方案的构建至关重要。 在`...

    android-testing-master.zip_Different_googlesamples

    "android-testing-master.zip" 是一个包含多种谷歌示例的资源库,它展示了自动化测试的各种框架和技术。这个压缩包旨在帮助开发者理解和掌握如何有效地测试Android应用。 一、Android测试概述 在Android平台上,...

    安卓混合脚本测试

    1. **JUnit**:作为Java的单元测试框架,JUnit是进行安卓应用单元测试的基础。开发者可以创建测试类,编写断言来验证代码逻辑的正确性。 2. ** Espresso**:Espresso是Google提供的一种UI测试工具,用于测试安卓...

    Android Test.rar

    这份2011年的面试题集虽然年代久远,但其涵盖的测试基础知识和理念对于现代Android开发者来说依然具有指导意义。掌握并理解这些知识点,将有助于你在面试中展现出深厚的技术功底,同时也能提升实际项目中的测试效率...

    android-testing-templates-master.zip

    1. **单元测试**(Unit Testing):这是测试的基础,主要针对应用程序的最小可测试单元,如方法或类。在Android中,JUnit和Mockito是常用的单元测试工具。JUnit用于编写测试用例,Mockito则可以帮助我们模拟对象的...

    AndroidTest:Android应用测试项目

    4. 持续集成(CI):自动化测试流程,及时发现并修复问题。 5. 代码覆盖率报告:使用JaCoCo等工具检查测试覆盖情况,确保测试的全面性。 6. 回归测试:每次修改代码后都要运行完整的测试套件,防止引入新问题。 7. ...

    BloomApp-test

    12. **持续集成/持续部署(CI/CD)**:如Jenkins、Travis CI等工具,可以自动化构建、测试和部署过程,确保每次代码更改后都能快速获取反馈。 13. **测试覆盖率**:通过JaCoCo等工具,可以测量测试覆盖了代码的多少...

Global site tag (gtag.js) - Google Analytics