JAVA单元测试用例编写
??单元测试通常不依赖第三方组件(比如数据库),如果遇到了这些需要依赖第三方的情况,可以使用mock技术,模拟想要的情况。Java中mock的对象主要是Java方法和Java类。PowerMock在Mockito的基础上扩展而来,支持Mockito的操作,是一个扩展了多个mock框架的、功能更加强大的框架。它使用一个自定义类加载器和字节码操作来模拟public方法(大多数情况)、static静态方法、构造方法、final类和方法、private私有方法、enum枚举类、去除静态初始化器等。
一、常用注解
- @RunWith(PowerMockRunner.class)
指定JUnit 使用 PowerMock 框架中的单元测试运行器来做Object的Mock工作,在执行时,再交给delegate的runner去执行。
- @PowerMockIgnore({}, {}, …)
默认情况下,PowerMock会用自己的类加载器去加载所有的类,除了system class(java.lang等目录下的类),使用PowerMockIgnore声明的类PowerMock也不会加载。
模拟final类、final方法或static静态方法时,需要添加@PrepareForTest注解,并指定方法所在的类。如果需要指定多个类,在{}中添加多个类并用逗号隔开即可。
- @Before、@BeforeClass、@After、@AfterClass
- @BeforeClass(@BeforeAll): 在测试类中只会被执行一次,注解在静态方法上。
- @Before(@BeforeEach):每个测试方法均会执行一次,注解在非静态方法上。
- 单元测试用例执行顺序: @BeforeClass -> @Before -> @Test -> @After -> @AfterClass。
- 测试方法的调用顺序:@Before -> @Test -> @After。
- 创建一个Mock实例,其所有属性均被置0或null,Mock函数默认不执行,有返回值的返回null、空集合或被对象类型包装的内置类型(Integer等)。
- Mock对象会记住所有的交互,即对这个Mock对象做的操作都被记录下来,然后可以选择性验证(verify方法) 感兴趣的交互。
@Component
public class TClass {
public String getMessage(String str) {
System.out.println(str);
return str + "!!!";
}
}
@Mock
private TClass tClass;
@Test
public void testTClass() {
tClass.getMessage("ho");
Mockito.verify(tClass).getMessage("ho");
}
- 创建一个没有Mock的实例,所有成员方法默认按照原方法的逻辑执行,直到被Mock返回某个具体的值为止。
- @Spy注解的变量需要被初始化,否则执行时会抛出异常。
- 如果发现@Spy注解修饰的变量是 null,会自动调用类的无参构造函数(必须有)来初始化,因此 new XXX() 可写可不写。
- 对Spy变量打桩时,使用 when 去设置模拟值,真实方法逻辑依然会执行,但模拟了返回结果。
创建一个可以调用真实方法逻辑的实例,并注入所有用@Mock或@Spy注解创建的实例。
在字段级别创建参数捕获器,在测试方法启动前(在@Before修饰的方法中),必须调用MockitoAnnotations.openMocks(this)进行初始化。
二、常用方法
2.1 Mockito类
org.mockito.Mockito类中提供了大量的静态方法,其中较多关键方法封装在静态类MockitoCore中。
2.1.1 mock和spy
public static <T> T mock(Class<T> classToMock) {
return mock(classToMock, withSettings());
}
public static <T> T mock(Class<T> classToMock, MockSettings mockSettings) {
return MOCKITO_CORE.mock(classToMock, mockSettings);
}
- 通过给定类或接口构建对应的mock实例。
- 对于非final类(接口、普通类等)的非final方法,不必使用@RunWith和@PrepareForTest注解,反之则相反。
public static <T> T spy(Class<T> classToSpy) {
return MOCKITO_CORE.mock(
classToSpy, withSettings().useConstructor().defaultAnswer(CALLS_REAL_METHODS));
}
public static <T> T spy(T object) {
return MOCKITO_CORE.mock(
(Class<T>) object.getClass(),
withSettings().spiedInstance(object).defaultAnswer(CALLS_REAL_METHODS));
}
- 可以只模拟对象的部分方法,同时其它方法跟原来一样,callRealMethod()方法也能实现部分mock。
- 通过模拟类的方式可直接调用静态方法。
- 通过when语句设置过的方法调用实际方法但返回模拟结果,没有通过when语句设置的方法调用原有方法并返回实际结果。
- 通常结合doXXX()方法,只有使用doReturn()方法时不会调用实际方法。
- spy对象可以视作被spy的对象的浅拷贝。
2.1.2
2.1.3
2.1.4 验证方法
2.2 ArgumentMatchers参数匹配
2.3 OngoingStubbing打桩
111
2.4 Assert类
public static <T extends Throwable> T assertThrows(Class<T> expectedThrowable,
ThrowingRunnable runnable) {}
允许使用异常类的超类型。
参考文档
《Mockito中文文档》
《Mockito官网》
《Mockito英文文档》
|