当前位置: 首页 > news >正文

使用J2MEUnit测试J2ME程序

作者:magic003 文章来源:http://www.j2medev.com/Article/ShowArticle.asp?ArticleID=903

JUnit是一个优秀的用于单元测试的框架,在j2se,j2ee的开发过程被广泛使用,它使得代码的质量得到更好的监控和维护。然而对于j2me平台一切就不是如此简单了,由于手持设备需要更多的考虑性能问题,j2me平台并没有提供反射(Reflectiong)的API,因此JUnit的很多功能无法在j2me上实现,程序员也就无法在j2me平台上使用JUnit进行单元测试。本文将介绍一个能对j2me应用程序进行单元测试的工具:J2MEUnit。j2meunit是一个基于JUnit实现的针对j2me平台的单元测试框架。详细介绍请看:http://j2meunit.sourceforge.net/

运行环境
本文的示例程序的运行和测试都是在eclipse3.1下进行,并需要到如下网址下载J2MEUnit的最新版本:
http://j2meunit.sourceforge.net/
下载后如为rar/zip压缩包,须先解压。

HelloWorld 实例
本文将通过一个HelloWorld的例子来介绍J2MEUnit的基本使用方法。首先在eclipse中新建一个J2ME Midlet Suite工程。File->New->Project->J2ME Midlet Suite,取名为HelloMidlet。
此工程包括两个类:HelloWorld.java和HelloWorldMidlet.java。首先在工程底下新建包名hello,然后在hello包底下创建一个新类:HelloWorld.java,和一个新的J2ME Midlet:HelloWorldMidlet.java。代码如下:

hello.HelloWorld.java

以下是引用片段:
packagehello;
publicclassHelloWorld{
publicHelloWorld(){
}

publicStringsayHello(){
return"HelloWorld";
}
}

hello.HelloWorldMidlet.java

以下是引用片段:
packagehello;
importjavax.microedition.lcdui.Alert;
importjavax.microedition.lcdui.AlertType;
importjavax.microedition.lcdui.Display;
importjavax.microedition.midlet.MIDlet;
importjavax.microedition.midlet.MIDletStateChangeException;
publicclassHelloWorldMidletextendsMIDlet{
privateDisplaydisplay;
privateAlertalert;
privateHelloWorldhello;

publicHelloWorldMidlet(){
display=Display.getDisplay(this);
hello=newHelloWorld();
}
protectedvoidstartApp()throwsMIDletStateChangeException{
Strings=hello.sayHello();
alert=newAlert("Hello",s,null,AlertType.INFO);
alert.setTimeout(Alert.FOREVER);
display.setCurrent(alert);
}
protectedvoidpauseApp(){
}
protectedvoiddestroyApp(booleanarg0)throwsMIDletStateChangeException{
}
}

程序的功能就是在一个Alert中输出“HelloWorld”的字符串,代码十分简单,在此不做过多论述。编译后,可以运行程序,如果运行后能在Alert中看到“HelloWorld”,说明程序运行正常。接着开始写测试代码。

编写测试类
写测试文件时需要用到J2MEUnit的类库,首先需要将类库加入到工程的build path中。右击工程名->properties->Java Build Path->Libraries, Add External JARs,选择刚才下载的文件,然后确定。
现在需要对HelloWorld.java中的sayHello()方法进行测试。在工程目录底下新建一个包名test,然后新建一个名为TestHelloWorld.java的类,主要对HelloWorld中的sayHello()方法进行测试。虽然J2MEUnit现在不支持反射机制,但是仍然按照JUnit的规范命名,以便日后的代码移植。TestHelloWorld代码如下所示:

TestHelloWorld.java

以下是引用片段:
packagetest;

importj2meunit.framework.Test;
importj2meunit.framework.TestCase;
importj2meunit.framework.TestMethod;
importj2meunit.framework.TestSuite;
importhello.HelloWorld;
publicclassTestHelloWorldextendsTestCase{
privateHelloWorldhello;

publicTestHelloWorld(){}

/**
*构造函数
*@paramsTestName测试方法的名称
*@paramrTestMethod测试的方法
*/
publicTestHelloWorld(StringsTestName,TestMethodrTestMethod){
super(sTestName,rTestMethod);
}
protectedvoidsetUp()throwsException{
super.setUp();
hello=newHelloWorld();
}

protectedvoidtearDown()throwsException{
super.tearDown();
}
publicTestsuite(){
TestSuiteaSuite=newTestSuite();

aSuite.addTest(newTestHelloWorld("testSayHello",newTestMethod(){
publicvoidrun(TestCasetc){
((TestHelloWorld)tc).testSayHello();
}
}));

returnaSuite;
}

/**
*测试sayHello()方法
*
*/
publicvoidtestSayHello(){
Strings=hello.sayHello();
assertEquals("HelloWorld!",s);
}
}

以下是引用片段:
packagetest;

importj2meunit.framework.Test;
importj2meunit.framework.TestCase;
importj2meunit.framework.TestMethod;
importj2meunit.framework.TestSuite;
importhello.HelloWorld;
publicclassTestHelloWorldextendsTestCase{
privateHelloWorldhello;

publicTestHelloWorld(){}

/**
*构造函数
*@paramsTestName测试方法的名称
*@paramrTestMethod测试的方法
*/
publicTestHelloWorld(StringsTestName,TestMethodrTestMethod){
super(sTestName,rTestMethod);
}
protectedvoidsetUp()throwsException{
super.setUp();
hello=newHelloWorld();
}

protectedvoidtearDown()throwsException{
super.tearDown();
}
publicTestsuite(){
TestSuiteaSuite=newTestSuite();

aSuite.addTest(newTestHelloWorld("testSayHello",newTestMethod(){
publicvoidrun(TestCasetc){
((TestHelloWorld)tc).testSayHello();
}
}));

returnaSuite;
}

/**
*测试sayHello()方法
*
*/
publicvoidtestSayHello(){
Strings=hello.sayHello();
assertEquals("HelloWorld!",s);
}
}

首先测试类必须从j2meunit.framework.TestCase继承。只是对HelloWorld类从测试,因此包含一个HelloWorld的对象。接着,重载了两个构造函数。同样,重写了TestCase的两个方法:setUp()和tearDown()。在setUp()中对hello进行了初始化。由于J2MEUnit不支持反射机制,因此我们必须自己写suite(),创建一个TestSuite的对象,然后加入要进行的测试方法。最后就是写测试sayHello()的方法,我们对sayHello()的返回值进行测试,看与预期结果是否匹配。
这是一个测试HelloWorld的类,我们还要写一个TestAll的类要运行测试过程。在test包下创建一个新类,取名TestAll,同样,也是继承自j2meunit.framework.TestCase,代码如下:

TestAll.java

以下是引用片段:
packagetest;
importj2meunit.framework.Test;
importj2meunit.framework.TestCase;
importj2meunit.framework.TestSuite;
importj2meunit.textui.TestRunner;
publicclassTestAllextendsTestCase{
publicTestAll(){
super("null");
}

publicTestAll(Stringname){
super(name);
}

/**
*@paramargs
*/
publicstaticvoidmain(String[]args){
String[]runnerArgs=newString[]{"test.TestHelloWorld"};
TestRunner.main(runnerArgs);
}
publicTestsuite(){
TestSuiteaSuite=newTestSuite();

aSuite.addTest(newTestHelloWorld().suite());

returnaSuite;
}
}

其中的main()方法是运行J2MEUnit的runner的,注意,其中的runnerArgs,是包含了各个测试类的名称的字符串数组,并且类名一定要带上完整的包名,否则运行会失败。

运行测试
好了,现在可以运行我们的测试了。右击TestAll,Run As..->Java Application。运行结束,可以在console窗口中看到如下输出:

以下是引用片段:
TestRunner.main()
.F
Time:0ms
FAILURES!!!
TestResults:
Run:1Failures:1Errors:0
Therewas1failure:
1)testSayHello(test.TestHelloWorld)"expected:〈HelloWorld!〉butwas:〈HelloWorld〉"

以下是引用片段:
TestRunner.main()
.F
Time:0ms
FAILURES!!!
TestResults:
Run:1Failures:1Errors:0
Therewas1failure:
1)testSayHello(test.TestHelloWorld)"expected:〈HelloWorld!〉butwas:〈HelloWorld〉"

因为我们在测试时,希望得到的结果是“Hello World!”,而实际返回结果是“Hello World”,所以会得到一个Failure。现在把“Hello World!”改成“Hello World”,再次运行,则会输出测试通过。Try it!

总结
这就是在J2ME平台上使用J2MEUnit进行单元测试的简单过程,更多的使用和API请参见J2MEUnit的文档。

相关文章:

  • 深入学习之图片预加载 和 闭包新认知
  • NetBeans 4.0创建J2ME手机RSS阅读器
  • J2ME游戏 之 NORMAL-FRAME
  • 深入学习之git操作
  • 开发NokiaS40系列应用程序初级篇
  • 深入学习之重写call方法
  • 深入学习之this 方法1
  • 基于Nokia S40的猜数字游戏之一
  • 深入学习之继承图总结
  • 基于Nokia S40的猜数字游戏之二
  • 深入学习之 this 方法2
  • 应用Nokia UI API处理声音问题
  • 深入学习之css重点
  • Nokia UI学习范例FishTank研究
  • react 向事件处理程序传递参数
  • JS 中的深拷贝与浅拷贝
  • 分享的文章《人生如棋》
  • 《Javascript数据结构和算法》笔记-「字典和散列表」
  • Flannel解读
  • java8-模拟hadoop
  • Js基础——数据类型之Null和Undefined
  • 理解 C# 泛型接口中的协变与逆变(抗变)
  • 爬虫模拟登陆 SegmentFault
  • 前端之Sass/Scss实战笔记
  • 区块链分支循环
  • 三栏布局总结
  • 深入浏览器事件循环的本质
  • 为物联网而生:高性能时间序列数据库HiTSDB商业化首发!
  • 消息队列系列二(IOT中消息队列的应用)
  • 一个6年java程序员的工作感悟,写给还在迷茫的你
  • 阿里云IoT边缘计算助力企业零改造实现远程运维 ...
  • ​​​​​​​sokit v1.3抓手机应用socket数据包: Socket是传输控制层协议,WebSocket是应用层协议。
  • ​【C语言】长篇详解,字符系列篇3-----strstr,strtok,strerror字符串函数的使用【图文详解​】
  • ​软考-高级-系统架构设计师教程(清华第2版)【第9章 软件可靠性基础知识(P320~344)-思维导图】​
  • ​软考-高级-信息系统项目管理师教程 第四版【第14章-项目沟通管理-思维导图】​
  • ​油烟净化器电源安全,保障健康餐饮生活
  • # 学号 2017-2018-20172309 《程序设计与数据结构》实验三报告
  • #include<初见C语言之指针(5)>
  • #NOIP 2014# day.2 T2 寻找道路
  • (1/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (二十五)admin-boot项目之集成消息队列Rabbitmq
  • (附源码)计算机毕业设计SSM保险客户管理系统
  • (南京观海微电子)——I3C协议介绍
  • (四)【Jmeter】 JMeter的界面布局与组件概述
  • (四)Android布局类型(线性布局LinearLayout)
  • (转)C#调用WebService 基础
  • (转)IOS中获取各种文件的目录路径的方法
  • (转)linux自定义开机启动服务和chkconfig使用方法
  • .a文件和.so文件
  • .NET/C# 异常处理:写一个空的 try 块代码,而把重要代码写到 finally 中(Constrained Execution Regions)
  • .net的socket示例
  • .net访问oracle数据库性能问题
  • .net解析传过来的xml_DOM4J解析XML文件
  • .so文件(linux系统)
  • [ C++ ] template 模板进阶 (特化,分离编译)