一、单元测试
单元测试是针对最小的功能单元编写的测试代码,对于Java来说最小的功能单元是方法,因此单元测试对于Java 来说就是针对单个Java方法的测试。
单元测试测试的基本思路是TDD(Test-Driven Development)测试驱动开发,通过测试驱动开发,提升开发质量。
单元测试的好处:
1、确保单个方法运行正常;
2、修改方法的代码后,单元测试可以快速自测;
3、可以自动化运行所有单元测试并获得报告;
4、单元测试代码可以直接用来演示方法的调用和执行;
二、Junit
Junit 当前版本:3.x / 4.x / 5.x
待测试的类及方法
public class Calculator{
public int calculate(String expression){
String[] ss = expression.split("\\+");
System.out.println(expression + " -> " + Arrays.toString(ss))
int sum = 0;
for(String s : ss){
sum = sum + Integer.parseInt(s);
}
return sum;
}
}
测试类及方法
public class CalculatorTest{
@Test
public void testCalculate(){
/*当未实现单元测试内容时,需要写fail 这样运行到这里的时候会执行失败,可以通过这种方式提示用户实现该单元测试内容
fail("Not yet implemented"); */
}
@Test
public void testCalculate(){
/*测试列举执行场景*/
assertEquals(3,new Calculator.calculate("1+2"));
assertEquals(6,new Calculator.calculate("1+2+3"));
assertEquals(35,new Calculator.calculate("12+23"));
}
@Test
public void testCalculate(){
/*测试执行异常输入场景,帮助方法编写考虑异常场景,实现测试驱动开发*/
assertEquals(3,new Calculator.calculate("1 +2"));
}
}
常用断言
断言判断 | assert |
断言相等 | assertEquals(100,x) |
断言数组相等 | assertArrayEquals({100,200,300},x) |
浮点数相等 | assertEquals(3.14159,x,0.0001) |
断言为 null | assertNull(x) |
断言为true/false | assertTrue(x>0) assertFalse(x>0) |
其他 | assertNotNull / assertNotEquals |
注意:
1、Junit 使用Assert 断言测试结果(浮点数assertEquals要制定delta);
2、每个测试方法必须完全独立;
3、测试代码必须非常简单;
4、不能为测试代码再编写测试;
5、方法输入场景考虑需要全面,才能起到单测效果;
三、常用注解
注解 | 解释 | 执行逻辑 | 常用场景 |
@Before | 初始化测试资源 | 在每一个测试方法执行前执行 | 初始化测试对象,例如 input = new FileInputStream(); |
@After | 释放测试资源 | 在每一个测试方法执行后执行 | 销毁@Before创建的测试对象,例如 input.close(); |
@BeforeClass | 初始化静态资源 | 在所有测试方法执行前执行 | 初始化非常耗时的资源,例如,创建数据库 |
@AfterClass | 释放静态资源 | 在所有测试方法执行后执行 | 清理@BeforeClass创建的资源,例如:删除数据库 |
以上注解Junit 执行逻辑
invokeBeforeClass(CalculatorTest.class); //@BeforeClass
for(Method testMethod : findTestMethods(CalculatorTest.class)){
CalculatorTest test = new CalculatorTest(); //new
test.setUp(); //@Before
testMethod.invoke(test);//@Test
test.tearDown();@After
}
invokeAfterClass(CalculatorTest.class);//@AfterClass
四、异常测试
/*测试代码执行若抛出 NumberFormatException 则测试通过;若抛出其他异常或者没有抛出异常,则测试不通过*/
@Test(expected = NumberFormatException.class)
public void testNumberFormatException(){
Integer.parseInt(null);
}
五、参数化测试
若待测试的输入和输出是一组数据
1、可以把测试数据组织起来;
2、用不同的测试数据调用相同的测试方法;
@RunWith(Parameterized.class)
public class AbsTest{
@Parameters
public static Collection<?> data(){
return Arrays asList(new Object[][]{{0,0},{1,1},{-1,1}});
}
int input ;
int expected;
public AbsTest(int input,int expected){
this.input = input;
this.expected = expected;
}
@Test
public void testAbs(){
int r = Math.abs(this.input);
assertEquals(this.expected, r);
}
}
参数必须由静态方法data() 返回,返回类型为Collection<Object[]>
静态方法必须标记为@Parameters 测试类必须标记为@RunWith(Parameterized.class) ,构造方法参数必须和测试参数对应。
六、超时测试
public class PItest{
PI pi = new PI();
/*timeout时间单位为毫秒,如下含义为不超过1000ms的情况的下,测试通过*/
@Test(timeout=1000)
public void testTimeout(){
/*计算pi到展开项的第1000个*/
double r = pi.calculate(1000);
assertEquals(3.14, r , 0.01);
}
}