JUNIT了解学习
转自:关于Java单元测试,你须要知道的一切html
转自: JUnit 入门教程java
JUnit高级用法之@RunWith
@RunWith
关于@RunWith注解,官方文档是这么描述的:spring
When a class is annotated with
@RunWith
or extends a class annotated with@RunWith
, JUnit will invoke the class it references to run the tests in that class instead of the runner built into JUnit.api
JUnit用例都是在Runner(运行器)来执行的。经过它,能够为这个测试类指定一个特定的Runner。那么大多数时候咱们都没有使用@RunWith这个注解,这是为何呢?其实,JUnit中有一个默认的Runner,它的名字叫BlockJunit4ClassRunner,但这是在JUnit4.4以后才引入的,对于4.4以前版本的JUnit,它的名字叫Junit4ClassRunner。在新版本的源代码中已经添加了注释来讲明这个问题:数组
/** * @deprecated Included for backwards compatibility with JUnit 4.4. Will be * removed in the next major release. Please use * {@link BlockJUnit4ClassRunner} in place of {@link JUnit4ClassRunner}. */ @Deprecated public class JUnit4ClassRunner extends Runner implements Filterable, Sortable { ...
写过关于Spring项目的单元测试的同窗可能见过这样的写法,就是用JUnit加载Spring的配置文件以完成Context的初始化,而后从Context中取出Bean并完成测试:app
import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.ApplicationContext; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(locations = { "classpath:applicationContext.xml" }) public class UserManagerTest { @Autowired ApplicationContext ctx; @Test public void testAddUser() { try { UserManager userManager = ctx.getBean(UserManager.class); userManager.addUser(); } catch (Exception e) { e.printStackTrace(); } } }
注意这里使用了@RunWith注解,代表这个类中的测试用例须要使用SpringJUnit4ClassRunner类来执行。框架
@RunWith(Suite.class)
其做用是使用JUnit执行一个测试套件。Suite类是JUnit自带的,意为套件,顾名思义,就是一套东西。经过它,能够把多个相关的测试类看作一个测试套件一块儿测试。看个例子:ide
import org.junit.runner.RunWith; import org.junit.runners.Suite; @RunWith(Suite.class) @Suite.SuiteClasses({ TestA.class, TestB.class, /*Any test class you want to run*/}) public class TestSuite { // Please note this case won't run. It will only run cases which // are configured in @Suite.SuiteClasses @Test public void testPrint() { System.out.println("Hello"); } }
@RunWith指定了Suite类,说明这个TestSuite类是一个套件。经过@Suite.SuiteClasses指定了要执行的测试类(这些类中的全部用例都会执行)。函数
须要注意的是,这个TestSuite类自己用例则不会执行了(如上面的testPrint()方法)。单元测试
@RunWith(Parameterized.class)
Parameterized类也是JUnit自带的,用于使用多个参数组合屡次执行同一个测试用例。看下面的例子:
import static org.junit.Assert.assertEquals; import java.util.Arrays; import java.util.List; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.Parameterized; import org.junit.runners.Parameterized.Parameters; @RunWith(Parameterized.class) public class TestParameterized { private int expected; private int first; private int second; public TestParameterized(int expected, int firstNumber, int secondNumber) { this.expected = expected; this.first = firstNumber; this.second = secondNumber; } /** * Note: @Parameters annotated method must be public static, * otherwise an Exception will thrown. */ @Parameters public static List<Integer[]> parameters() { return Arrays.asList(new Integer[][]{{3, 1, 2}, {5, 2, 3}, {7, 3, 4}, {9, 4, 5}}); } @Test public void testAdd() { String format = "Using parameters: expect=%d, first=%d, second=%d"; System.out.println(String.format(format, expected, first, second)); Feature feature = new Feature(); assertEquals(expected, feature.add(first, second)); } @Test public void testPrint() { String format = "Print ----------: expect=%d, first=%d, second=%d"; System.out.println(String.format(format, expected, first, second)); } } class Feature { public int add(int i1, int i2) { return i1 + i2; } }
执行结果以下:
能够看到,虽然TestParameterized类中只有两个测试用例testAdd和testPrint,可是结果输出了8行,即每一个用例都执行了4遍。
使用Parameterized注解须要注意几点:
- 该方法要有构造函数
- 有一个public static的方法被@Parameters标注,而且该方法只能返回Iterable类型或数组类型的数据(源代码是以下处理的)
if (parameters instanceof Iterable) { return (Iterable<Object>) parameters; } else if (parameters instanceof Object[]) { return Arrays.asList((Object[]) parameters); } else { throw parametersMethodReturnedWrongType(); }
由于上面的方式使用了构造方法来初始化数据,其实也可使用字段注入来代替构造方法,只需稍加改变TestParameterized类便可:
- 用Parameter参数来修饰属性。注意:索引从0开始
- 属性要用public修饰
@Parameter(0) public int expected; @Parameter(1) public int first; @Parameter(2) public int second;
@RunWith(Categories.class)
顾名思义,执行一个“类别”。和Suite相似,只是Suite是执行指定类中的全部用例,而Categories执行的范围更小,是在Suite的基础上只执行指定的“类别”的用例。这就须要事先在各个测试用例上用@Category标注该用例属于那些“类别”,以后即可以经过类别来选择执行某些用例。看例子:
/*-----TestA.java-----*/ import org.junit.Test; import org.junit.experimental.categories.Category; class Feature1 {} class Feature2 {} public class TestA { @Test @Category(Feature1.class) public void testAdd() { System.out.println("A.testAdd"); } @Test @Category(Feature2.class) public void testAdd2() { System.out.println("A.testAdd2"); } @Test @Category({Feature1.class, Feature2.class}) public void testAdd3() { System.out.println("A.testAdd3"); } } /*-----TestCategory.java-----*/ import org.junit.experimental.categories.Categories; import org.junit.experimental.categories.Categories.ExcludeCategory; import org.junit.experimental.categories.Categories.IncludeCategory; import org.junit.runner.RunWith; import org.junit.runners.Suite; @RunWith(Categories.class) @IncludeCategory(Feature1.class) @ExcludeCategory(Feature2.class) @Suite.SuiteClasses({ TestA.class, /*Any test class you want to run*/}) public class TestCategory { // Do nothing }
其中,Feature1和Feature2表明两个不一样的“类型”,TestA类中经过@Category标注了各个用例(能够为一个用例指定多个Category,例如上方的testAdd3方法)。@IncludeCategory指明了须要执行的类型,而@ExcludeCategory指明了不但愿执行的类型,这个注解对于过滤相似testAdd3这样有多个类型的用例颇有效。如下是执行结果:
能够看到,只有标注了Feature1的用例执行了,并且带有Feature2的则被过滤掉了,因此只剩下testAdd这一个用例了。
@RunWith(Theories.class)
提供一组参数的排列组合值做为待测方法的输入参数。同时注意到在使用Theories这个Runner的时候,咱们的待测方法能够拥有输入参数,而这在其它的Runner中的测试方法是不行的。下面是一个例子:
import org.junit.experimental.theories.DataPoint; import org.junit.experimental.theories.Theories; import org.junit.experimental.theories.Theory; import org.junit.runner.RunWith; @RunWith(Theories.class) public class TestTheories { @DataPoint public static String nameValue1 = "Tony"; @DataPoint public static String nameValue2 = "Jim"; @DataPoint public static int ageValue1 = 10; @DataPoint public static int ageValue2 = 20; @Theory public void testMethod(String name, int age){ System.out.println(String.format("%s's age is %s", name, age)); } }
上面的代码的意思是,将”Tony”、”Jim”、十、20四个参数以类型合法的排列组合传给待测方法。所以输出的结果必然也有2x2=4种。下面是执行结果:
不过,为了简单,咱们除了可使用@DataPoint注解来提供参数以外,还能够经过@DataPoints注解来提供参数,参照上述代码,只须要将@DataPoint注解标注的四个字段参数替换为以下的两个便可:
@DataPoints public static String[] names = {"Tony", "Jim"}; @DataPoints public static int[] ageValue1 = {10, 20};
总结
介绍了这么几种Runner,如今回过头来看看一开始提到的SpringJUnit4ClassRunner,其实这个类继承与JUnit默认的运行器BlockJUnit4ClassRunner,来看看源代码中的声明(官方文档):
public class SpringJUnit4ClassRunner extends BlockJUnit4ClassRunner {
继承的好处就是能够彻底保留默认的功能,而且提供了一套支持Spring上下文的框架,正如官方文档所说:
SpringJUnit4ClassRunner
is a custom extension of JUnit'sBlockJUnit4ClassRunner
which provides functionality of the Spring TestContext Framework to standard JUnit tests by means of theTestContextManager
and associated support classes and annotations.