Junit 单元测试
JUnit 是一个 Java 编程语言的单元测试框架。JUnit 在测试驱动的开发方面有很重要的发展,是起源于 JUnit 的一个统称为 xUnit 的单元测试框架之一
Junit 常用注解:
@Test:表示方法是测试方法
@ParameterizedTest :表示方法是参数化测试
@RepeatedTest :表示方法可重复执行
@DisplayName :为测试类或者测试方法设置展示名称
@BeforeEach :表示在每个单元测试之前执行
@AfterEach :表示在每个单元测试之后执行
@BeforeAll :表示在所有单元测试之前执行
@AfterAll :表示在所有单元测试之后执行
@Tag :表示单元测试类别,类似于JUnit4中的@Categories
@Disabled :表示测试类或测试方法不执行,类似于JUnit4中的@Ignore
@Timeout :表示测试方法运行如果超过了指定时间将会返回错误
@ExtendWith :为测试类或测试方法提供扩展类引用
Junit5 新的特性
1)异常断言 :Assertions.assertThrows()
2)超时断言:Assertions.assertTimeout()
3)重复测试:@RepeatedTest注解,允许某个单元测试执行多次
4) 动态测试:允许动态的创建单元测试,通过@TestFactory注解,会在运行时生成单元测试
5) 参数化测试:利用@ValueSource等注解,指定入参,我们将可以使用不同的参数进行多次单元测试,而不需要每新增一个参数就新增一个单元测试,省去了很多冗余代码。
@ValueSource: 为参数化测试指定入参来源,支持八大基础类以及String类型,Class类型
@NullSource: 表示为参数化测试提供一个null的入参
@EnumSource: 表示为参数化测试提供一个枚举入参
@CsvFileSource:表示读取指定CSV文件内容作为参数化测试入参
@MethodSource:表示读取指定方法的返回值作为参数化测试入参
Junit 使用案例
1)引入Junit 包,修改项目pom文件
2)编写单元测试方法
3)执行测试
PowerMockito 单元测试
PowerMockito是一个扩展了其它如EasyMock等mock框架的、功能更加强大的框架。PowerMockito 使用一个自定义类加载器和字节码操作来模拟静态方法、构造方法、final类和方法、私有方法、去除静态初始化器等等
PowerMockito 常用注解:
@RunWith(PowerMockRunner.class) 使用PowerMockRunner进行测试
@PrepareForTest({ExtensionLoader.class,Configs.class}) 静态类,测试类本身(需要mock构造方法时)
@PowerMockIgnore("javax.management.*") 为了解决使用powermock后,提示classloader错误
Mock常用注解:
@Mock: 创建一个Mock.
@InjectMocks: 创建一个实例,简单的说是这个Mock可以调用真实代码的方法,其余用@Mock(或@Spy)注解创建的mock将被注入到用该实例中。
PowerMockito使用案例
1)引入PowerMockito包,修改项目pom文件
2)使用示例
(a): 普通Mock: Mock参数传递的对象
Powermockito.mock():主要是通过class创建对应的mock对象,不同于easymock等使用proxy(代理)的方式创建,而是在运行过程中动态修改class字节码文件的形式来创建。
Do..when..then:可以理解为,就是在什么时候,做了什么事,然后怎么样了。
Verify:可以理解为验证无返回值时候,查看是否被调用。对于无返回值类型的测试,只能验证其是否被调用
(b): Mock方法内部new出来的对象
PowerMockito.whenNew(StudentDao.class).withNoArguments().thenReturn(studentDao);//构建无参对象
当使用PowerMockito.whenNew方法时,必须加注解@PrepareForTest和@RunWith。注解@PrepareForTest里写的类是需要mock的new对象【mock - final方法/类同理】代码所在的类。
@PrepareForTest(StudentFinalDao.class) 中的类为 final类 或final方法所在的类
(c): Mock普通类的静态方法
(e): Mock系统类的静态和final方法
(f): 模拟THREAD.SLEEP()时抛出中断异常的场景
PowerMock简单实现原理
• 当某个测试方法被注解@PrepareForTest标注以后,在运行测试用例时,会创建一个新的org.powermock.core.classloader.MockClassLoader实例,然后加载该测试用例使用到的类(系统类除外)。
• PowerMock会根据你的mock要求,去修改写在注解@PrepareForTest里的class文件(当前测试类会自动加入注解中),以满足特殊的mock需求。例如:去除final方法的final标识,在静态方法的最前面加入自己的虚拟实现等。
• 如果需要mock的是系统类的final方法和静态方法,PowerMock不会直接修改系统类的class文件,而是修改调用系统类的class文件,以满足mock需求。
3)Mock 异常
a. mock 异常的发生
Mockito.when(Mockito.anyObject()) .thenThrow(new Exception());
Mockito.when(Mockito.anyObject()) .thenThrow(Exception.class);
b.对待无返回值的方法
使用PowerMockito.doThrow()方法.doThrow放的是需要模拟的异常,when()放的是产生异常的对象,括号外面写对象调用的方法
PowerMockito.doThrow(IllegalStateException.class) .when(dictMock) ....;
c. toThrow mock异常对象