JUnit 4是JUnit框架有史以来的最大改进,其主要目标便是利用Java 5的Annotation特性简化测试用例的编写。让我们看看如何使用JUnit 4来进行Unit测试。 请注意:本文主要介绍JUnit 4的最新特性和用法,并不会讲解Unit测试的基础。 我们使用的开发环境是Eclipse 3.2,它已经自带了JUnit 4.1,你需要将JUnit 4 Library添加到项目用到的Library中。另外,必须使用JDK 5.0或更高版本。 要在Eclipse环境之外运行JUnit,需要下载JUnit 4.1,具体请访问JUnit.org。 我们先看一个简单的Math类:
package
com.javaeedev.junit4;
public class Math { public int abs(int value) { return value>=0 ? value : (- value); } public int div(int a, int b) { return a / b; } /** * BUG: if b less than 0! */ public float exp(int a, int b) { float r = 1 ; for(int i=0; i<b; i++ ) r = r * a; return r; } } 注意exp()方法是有Bug的,如果传入参数2, -1,则期待的返回值应为0.5F,但实际返回值为1.0F。 下面我们看看传统的JUnit的TestCase:
public class MathTest extends
TestCase {
public void setUp() { super .setUp(); } public void tearDown() { super .tearDown(); } public void testAbs() { assertTrue(true ); } public void testDiv() {...} public void testExp() {...} } JUnit依赖反射来执行每个以test开头的方法。然而,在最新的JUnit 4中,由于有了Annotation的支持,我们的测试方法不需要再以testXxx标识了,而是写上一个@Test标注即可。例如:
@Test public void doAbs() {...}
甚至MathTest类也不必继承自TestCase。你也许会想到,不继承自TestCase就无法调用assertXxx方法了,正因为如此,所有的assertXxx方法全部以静态方法被放入了Assert类,使用Assert.assertXxx()调用。如果使用
import static org.junit.Assert.*;
则原有的代码不必改动。 setUp()和tearDown()方法也依赖@Before和@After标记,这样做的最大的好处是在继承体系内不必担心忘记了在setUp()方法中调用父类的super.setUp()方法,JUnit框架会自动处理父类的@Before和@After标记的方法。 并且,JUnit框架对@Before和@After的调用顺序类似于类的构造方法和析构方法,即@Before按照父类到子类的顺序调用,@After则相反,这样保证了资源的正确获取和释放。 当然,不再强迫必须使用setUp和tearDown作为方法名,可以使用更有意义的方法名,例如:initDatabase()和closeDatabase(),只要它们被标注了@Before和@After即可。 来看看使用Annotation的MathTest:
package
com.javaeedev.junit4;
import static org.junit.Assert.* ; import org.junit.* ; public class MathTest { public MathTest() { System.out.println("new MathTest instance." ); } @Before public void setUp() throws Exception { System.out.println("call @Before before a test method" ); } @After public void tearDown() throws Exception { System.out.println("call @After after a test method" ); } @Test public void doAbs() { Math math = new Math(); assertEquals(200, math.abs(200 )); assertEquals(100, math.abs(-100 )); assertEquals(0, math.abs(0 )); } @Test public void doDiv() { Math math = new Math(); assertEquals(5, math.div(100, 20 )); assertEquals(4, math.div(100, 21 )); } @Test(expected=ArithmeticException.class ) public void doDiv0() { new Math().div(127, 0 ); } @Test(timeout=1 ) public void doLongTimeTask() { double d = 0 ; for(int i=1; i<10000000; i++ ) d+= i; } @Test public void testExp() { Math math = new Math(); assertEquals(32f, math.exp(2, 5), 0.001f ); assertEquals(1f, math.exp(2, 0), 0.001f ); assertEquals(0.5f, math.exp(2, (-1)), 0.001f ); } } 对测试异常,JUnit 4可以用expected=Exception.class来期待一个预期的异常,而不必编写
try
{
... fail("No exception" ); } catch (Exception e) { // OK! } 来看看doDiv0测试,我们期待一个除数为0的ArithmeticException,因此编写如下测试方法:
@Test(expected=ArithmeticException.class
)
public void doDiv0() { new Math().div(127, 0 ); } 对于非常耗时的测试,@Test还有一个timeout来标识该方法最长执行时间,超过此时间即表示该测试方法失败:
@Test(timeout=1
)
public void doLongTimeTask() { double d = 0 ; for(int i=1; i<10000000; i++ ) d+= i; } 以上方法若执行时间超过1ms则测试失败,由于依赖CPU的执行速度,在不同的机器上测试结果也不同。 JUnit 4另一个较大的变化是引入了@BeforeClass和@AfterClass,它们在一个Test类的所有测试方法执行前后各执行一次。这是为了能在@BeforeClass中初始化一些昂贵的资源,例如数据库连接,然后执行所有的测试方法,最后在@AfterClass中释放资源。 正如你能想到的,由于@BeforeClass和@AfterClass仅执行一次,因此它们只能标记静态方法,在所有测试方法中共享的资源也必须是静态引用:
private static
Object dbConnection;
@BeforeClass public static void setUpBeforeClass() throws Exception { System.out.println("call @BeforeClass and init database connection" ); dbConnection = new Object(); } @AfterClass public static void tearDownAfterClass() throws Exception { System.out.println("call @AfterClass to release database connection" ); dbConnection = null ; } |
JUnit4快速入门(转)
最新推荐文章于 2024-09-14 23:01:09 发布
JUnit4快速入门(转)
2008-08-06 10:47