收藏 分享(赏)

android上的单元测试.doc

上传人:HR专家 文档编号:5964968 上传时间:2019-03-22 格式:DOC 页数:9 大小:227.50KB
下载 相关 举报
android上的单元测试.doc_第1页
第1页 / 共9页
android上的单元测试.doc_第2页
第2页 / 共9页
android上的单元测试.doc_第3页
第3页 / 共9页
android上的单元测试.doc_第4页
第4页 / 共9页
android上的单元测试.doc_第5页
第5页 / 共9页
点击查看更多>>
资源描述

1、Android 上的单元测试【IT168 技术文档】任何程序的开发都离不开单元测试来保证其健壮和稳定。Android 的程序自然也不例外。从 Android SDK 0.9 开始,就有了比较成熟的测试框架,但是直到目前最新的 1.1 版本,也没有详细的文档介绍这个内容,只是简单的给了一个 Api Demos 里的几个单元测试代码。因此,我在这里对此内容做一下梳理和总结:JUnit 还能用么?在 Java 下做单元测试必然用到 JUnit。这里说的 JUnit 是指从 Apache 基金会下载的junit.jar 里提供的一系列单元测试功能。这些功能显然是运行在 JDK 之上的。在 Androi

2、d下已经没有了 JDK,自然也无法运行 JUnit。但是这并不妨碍我们利用 JUnit 编写单元测试。只不过在运行单元测试时,一定要用 JDK 来运行,利用 java 命令来启动 JUnit 的某个Runner。如果是用 Eclipse 的话,可以在 Run Configuration 里新建一个 JUnit。但是一定要记得在 Classpath 选项卡里将 Bootstrap Entries 中的 Android Library 改成 JRE,并且添加 junit.jar。很明显的,这种测试就是正规的 Java 单元测试,和 Android 没有任何关系。你无法测试任何关于 Android

3、系统中的 API,你写的 Activity,人机界面等等。所以,如果你想测试仅仅是一些封装数据的对象,或者是纯粹的数值计算,还是可以用这种方法的。Android 里面的 junit.framework 包是怎么回事?很多人看到这个包的时候,第一反应是 Android 是不是已经完整集成了 JUnit。很遗憾这不是事实。如果你按照 JUnit 的运行方法,却不像上面那样改用 JDK,就一定会得到一个异常:# An unexpected error has been detected by Java Runtime Environment:# Internal Error (classFilePa

4、rser.cpp:2924), pid=4900, tid=4476#Error: ShouldNotReachHere()# Java VM: Java HotSpot(TM) Client VM (10.0-b19 mixed mode windows-x86)# An error report file with more information is saved as:# E:MydocEclipseWorkspaceTestAndroidhs_err_pid4900.log# If you would like to submit a bug report, please visit

5、:#http:/ 这个类用于在 Android 担当所有独特的 TestCase 的基类的作用,它是一个 Abstract Class。Android 单元测试类继承关系图如下所示:之所以有那么多 XXXTestCase 主要是为了简化工作。例如当你想对一个访问数据库的功能进行测试时,首先需要自己启动并初始化数据库。在这里是类似的,如果你想测试一个 Activity,首先要启动它。而 ActivityTestCase 就会自动帮你做完这些事情。而 ActivityUnitTestCase 会更注重测试的独立性,它会让测试与 Android 底层的联系降到最低。其余的类可以查看相关的 Javad

6、oc 来按需挑选。要编写测试,就是找到合适的XXXTestCase 作为基类来继承,并且编写自己的测试方法。很明显的,最简单的编写测试的方法就是继承 AndroidTestCase 写一个自己的TestCase。然后为自己的一组 TestCase 写一个 Activity 界面,由界面控制 TestCase 的启动,运行和结果报告。但是,你很快会发现,为何要给测试写一个界面呢?这太诡异了。这时就需要一种技术,它可以利用命令行(Shell)来启动一组测试,并且通过命令行的形式给出结果。这就是所谓的 Instrumentation。什么是 Instrumentation?一般在开发 Android

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

8、s for implementing application 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 descr

9、ibed to the system through an AndroidManifest.xmls tag.“对于单元测试,我们需要认真了解的就是 android.test.InstrumentationTestRunner 类。这是 Android 单元测试的主入口。它相当于 JUnit 当中 TestRunner 的作用。那么如何加载它呢,首先要在 manifest 文件中加入一行关于 Instrumentation 的声明。比如 Android Api Demos 中的测试里的 manifest 是这么写的(我滤掉了所有的注释) :如果用 Eclipse 的 ADT 插件(0.8 版本

10、以上),也可以用图形界面来添加,如下图:编辑好 manifest,就可以打包(build ,可以用 Eclipse ADT 来做,也可以用 aapt 命令手工完成),然后安装到虚拟机上 (用 adb install 命令)。之后就可以利用命令行的方式来加载你的单元测试了。在 Android Shell 中加载一个 Instrumentation 的方法是利用以下命令:adb shell am instrument w XXXXXX其中-w 是指定 Instrumentation 类的参数标志。一个简单的例子是:adb shell am instrument -w com.android.foo

11、/android.test.InstrumentationTestRunner当然,也可以利用 adb shell 先进入 android 命令行模式,再直接写 am instrument w XXXXXXX。下面将具体介绍如何将根据需要加载一组单元测试。如何在 Android 中利用 Instrumentation 来进行测试?在介绍具体的命令之前,我们先理解一下单元测试的层次。一组单元测试可以被组织成若干个 TestSuite。每个 TestSuite 包含若干 TestCase(某个继承 android.jar 的junit.framework.TestCase 的类)。每个 TestC

12、ase 又包含若干个 Test(具体的 test 方法)。如果假设 com.android.foo 是你的测试代码的包的根。当执行以下命令时,会执行所有的 TestCase 的所有 Test。测试的对象就是在 Target Package 中指定的包中的代码:adb shell am instrument -w com.android.foo/android.test.InstrumentationTestRunner如果你想运行一个 TestSuite,首先继承 android.jar 的 junit.framework.TestSuite 类,实现一个 TestSuite(比如叫 com.

13、android.foo.MyTestSuite),然后执行以下命令执行此 TestSuiteadb shell am instrument -e class com.android.foo.MyTestSuite -w com.android.foo/android.test.InstrumentationTestRunner其中的-e 表示额外的参数,语法为-e arg1 value1 arg2 value2 这里用到了 class参数。如果仅仅想运行一个 TestCase(比如叫 com.android.foo.MyTestCase),则用以下命令:adb shell am instrum

14、ent -e class com.android.foo.MyTestCase -w com.android.foo/android.test.InstrumentationTestRunner如果仅仅想运行一个 Test(比如就是上面 MyTestCase 的 testFoo 方法) ,很类似的,就这样写:adb shell am instrument -e class com.android.foo.MyTestCase#testFoo -w com.android.foo/android.test.InstrumentationTestRunner然后,所有的测试结果会输出到控制台,并会

15、做一系列统计,如标记为 E 的是 Error,标记为 F 的是 Failure,Success 的测试则会标记为一个点。这和 JUnit 的语义一致。如果希望断点调试你的测试,只需要直接在代码上加上断点,然后将运行命令参数的-e 后边附加上 debug true 后运行即可。更加详细的内容可以看 InstrumentationTestRunner 的 Javadoc。我希望 Android 能尽快有正式的文档来介绍这个内容。如何在 Android 的单元测试中做标记?在 android.test.annotation 包里定义了几个 annotation,包括 LargeTest,Medium

16、Test,SmallTest,Smoke,和Suppress 。你可以根据自己的需要用这些 annotation 来对自己的测试分类。在执行单元测试命令时,可以在-e 参数后设置“size large”/ “size medium”/ “size small”来执行具有相应标记的测试。特别的Supperss 可以取消被标记的 Test 的执行。完整的操作过程总结以上所有的内容,编写并运行完整的测试需要以下的步骤:以上步骤中,在 Android 自带的例子中,我发现它有两个 manifest.xml。也就是说在步骤 3 中源代码和测试代码分别生成了两个不同的包。然后步骤 4 利用 adb ins

17、tall 命令安装到了虚拟机上。由于我没有找到 Eclipse ADT 有办法可以为一个只有 Instrumentation,没有Activity 的 Application 打包并安装,于是采用了略微不同的办法完成了这个工作。下文中将一一详细介绍整个过程。1、 编写程序我新建了一个项目 TestApp,参数为:Package Name: com.android.testappActivity Name: MainActivityApplication Name: TestApp以下是 MainActivity 的源代码:packagecom.android.testapp;importand

18、roid.app.Activity;importandroid.os.Bundle;publicclassMainActivityextendsActivity /* Called when the activity is first created. */OverridepublicvoidonCreate(Bundle savedInstanceState) super.onCreate(savedInstanceState);setContentView(R.layout.main);publicintsum(inta,intb) returna + b;publicintsubstra

19、ct(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 and

20、roid.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);MediumTestpublic void testS

21、um() assertEquals(3, getActivity().sum(1, 2);MediumTestpublic void testSubstract() assertEquals(-1, getActivity().substract(1, 2);我继承了 ActivityInstrumentationTestCase。这个 TestCase 在执行时会自动帮我启动相应的 Activity。接下来就是程序的 Manifest:在这个文件中,我将 Activity 和 Instrumentation 的声明写到了一起,而没有像 Apis Demo 那样分开。请注意里面的标签。如果没有

22、那句,在运行测试时会报告找不到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 class com.android.testapp.test.TestMainActivity w com.android.testapp/android.test.InstrumentationTestRunner即可看到如下的结果:可以看到,单元测试正确的找到了减法中的错误。结果中的成功的测试显示为”.”,一个失败的显示为”F” 。只不过我还是不太理解为什么我只写了两个测试方法,Tests run 却显示了 3。

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 企业管理 > 经营企划

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报