JUnit5技术分享

简介: Spring Boot 2.2.0 版本开始引入Junit5 作为单元测试默认库最新的JUnit框架 与之前的有很大不同由三个不同子项目的几个不同模块组成Junit Platform (公共测试平台): 基础核心内容 ,不包括单元测试的一些测试引擎JUnit Junpiter(核心测试引擎): 是JUnit5 新特性的核心。内部包含了一个测试引擎,用于在JunitPlatform上运行JUnit Vintage:兼容JUnit4 JUnit3

一,JUnit5常用注解



https://junit.org/junit5/docs/current/user-guide/#writing-tests-annotations


1.@DisplayName


1.png


package com.yer.boot;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
@DisplayName("junit5 功能测试类")
public class JUnit5Test {
    @DisplayName("测试displayname注解")
    @Test
    void testDisplayName(){
        System.out.println(999);
    }
}


2.@BeforeEach


@AfterEach


@BeforeAll


@AfterAll


package com.yer.boot;
import org.junit.jupiter.api.*;
@DisplayName("junit5 功能测试类")
public class JUnit5Test {
    @DisplayName("测试displayname注解")
    @Test
    void testDisplayName() {
        System.out.println(999);
    }
    @DisplayName("测试displayname注解2")
    @Test
    void testDisplayName2() {
        System.out.println(9992);
    }
    @BeforeEach
        //每个单元测试之前都要执行
    void testBeforeEach() {
        System.out.println("测试要开始了!");
    }
    @AfterEach
        //每个单元测试之后都要执行
    void testAfterEach() {
        System.out.println("测试结束了");
    }
    @BeforeAll
    static void testBeforeAll() {
        System.out.println("要测试所有测试了");
    }
    @AfterAll
    static void testAfterAll() {
        System.out.println("所有测试测试完了");
    }
}


3.@Tag 表示单元测试类别,类似于JUni4中的@Categories

4.@Disable 不用执行


2.png


5.@Timeout


/**
*
* 规定方法的超时时间
* 超出时间测试异常
* @throws InterruptedException
*/
@Timeout(value = 500,unit = TimeUnit.MILLISECONDS)
    @Test
    void testTimeout() throws InterruptedException {
        Thread.sleep(1000);
    }


6.@ExtendWIth:为测试类或测试方法提供扩展类引用 (类似于junit4@RunWith)


@SpringBootTest 复合注解中有@ExtendWIth(SpringExtension.class)


7.@RepeatTest(8) 重复测试


二,断言assertions



断言是测试方法中的核心部分,用来对测试需要满足的条件进行验证


这些断言方法都是org.junit.jupiter.api.Assertions中的静态方法


JUnit5 内置的断言可以分成以下几个类别:检查业务逻辑返回的数据是否合理


得益于断言机制,所有的测试结束之后会有一个详细报告


1.简单断言


对单个值进行简单验证


 @DisplayName("测试简单断言")
    @Test
    void testSimpleAssertions(){
        int cal = cal(3,3);
        assertEquals(6,cal);
    }
    int cal(int i,int j){
        return i+j;
    }
//成功 
=======================================
    /**
     *
     * 断言
     * 前面断言失败,后面代码不会执行
     */
    @DisplayName("测试简单断言")
    @Test
    void testSimpleAssertions(){
        int cal = cal(2,3);
        assertEquals(6,cal,"业务逻辑计算失败");
        Object o1 = new Object();
        Object o2 = new Object();
        assertEquals(o1,o2,"两个对象不一样");
    }
    int cal(int i,int j){
        return i+j;
    }


2.数组断言


来判断两个对象或者原始类型的数组是否相等


@Test
@DisplayName("array assertion")
public void array(){
    assertArrayEquals(new int[]{1,2},new int{1,2});
}


3.组合断言


assertAll方法接收多个org.junit.jupiter.api.Executable函数式接口的实例作为要验证的断言,可以通过lambda表达式 很容易的提供这些断言


 @Test
 @DisplayName("组合断言")
  void all(){
    assertAll("test",
              ()->assertTrue(true&&true,"结果不为true"),
              ()->assertEquals(1,1,"结果不为1"));
    system.out.println("两个断言都成功才可以输出")
  }


4.异常断言

在junit4时期 ,想要检测方法异常情况时,需要用注解@Rule注解的@ExpectedException变量还是比较麻烦的。相对于现在的Junit5提供了一种新的断言方式Assertions.assertThrows(),配合函数式编程就可以进行使用


断定业务逻辑一定出现异常


@Test
    @DisplayName("异常断言")
    void testException(){
        assertThrows(ArithmeticException.class,
                ()-> { int i = 1/0;},"业务逻辑竟然能正常运行,不是吧");
    }


5.超时断言


还提供了Assertions.assertTimout()为测试方法设置了超时时间


@Test
@DisplayName("超时断言")
public void timeOut(){
    //如果测试方法时间超过1s
    Assertions.assertTimeout(Duration.ofMillis(1000),
                             ()->Thread.sleep(500));
}

6.快速失败


通过fail方法直接使得测试失败


 

    @Test
    @DisplayName("快速失败")
    void testFai(){
        if (2 == 2 ){
            fail("测试失败");
        }
    }


三,前置条件(assumptions)



JUnit5中的前置条件(assunption假设)类似于断言,不同之处在于不满足的断言会使得测试方法失败,而不满足的前置条件只会使得测试方法执行终止。


前置条件爱你可以堪称是测试方法执行的前提,当该前提不满足时,就没有继续执行的必要l


 

    @DisplayName("测试前置条件")
    @Test
    void testAsummptions(){
        Assumptions.assumeTrue(false,"结果不是true");
        System.out.println("为true");
    }

disable 跳过


假设失败跳过


四,嵌套测试



-----------可参考官方文档


junit可以通过加入了嵌套测试


嵌套测试情况下


外层的Test不能驱动内层的Before(After)Each/All主类的方法提前/之后运行


内层的可以驱动外层的


package com.yer.boot;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
import org.junit.jupiter.api.Test;
import java.util.EmptyStackException;
import java.util.Stack;
import static org.junit.jupiter.api.Assertions.*;
@DisplayName("嵌套测试")
public class TestAStackDemo {
    Stack<Object> stack;
    @Test
    @DisplayName("new Stack()")
    void isInstantiatedWithNew() {
        new Stack<>();
    }
    @Nested
    @DisplayName("when new")
    class WhenNew {
        @BeforeEach
        void createNewStack() {
            stack = new Stack<>();
        }
        @Test
        @DisplayName("is empty")
        void isEmpty() {
            assertTrue(stack.isEmpty());
        }
        @Test
        @DisplayName("throws EmptyStackException when popped")
        void throwsExceptionWhenPopped() {
            assertThrows(EmptyStackException.class, stack::pop);
        }
        @Test
        @DisplayName("throws EmptyStackException when peeked")
        void throwsExceptionWhenPeeked() {
            assertThrows(EmptyStackException.class, stack::peek);
        }
        @Nested
        @DisplayName("after pushing an element")
        class AfterPushing {
            String anElement = "an element";
            @BeforeEach
            void pushAnElement() {
                stack.push(anElement);
            }
            @Test
            @DisplayName("it is no longer empty")
            void isNotEmpty() {
                assertFalse(stack.isEmpty());
            }
            @Test
            @DisplayName("returns the element when popped and is empty")
            void returnElementWhenPopped() {
                assertEquals(anElement, stack.pop());
                assertTrue(stack.isEmpty());
            }
            @Test
            @DisplayName("returns the element when peeked but remains not empty")
            void returnElementWhenPeeked() {
                assertEquals(anElement, stack.peek());
                assertFalse(stack.isEmpty());
            }
        }
    }
}


五,参数化测试



可参考官方文档


参数化测试是Junit5 很重要的一个新特性


它使得用不同的参数多次运行测试成为了可能,也为我们的单元测试带来许多便利。


利用**@ValueSource**等注解,指定入参,我们将可以使用不同的参数进行多次单元测试,而不需要每新增一个参数就新增一个单元测试,省去了很多冗余代码。


  • @ValueSource:为参数化测试指定入参来源,支持八大基础类 以及String类型Class类型


  • @NullSource表示为参数化测试提供一个null的入参@EnumSource表示为参数化测试提供一个枚举入参


  • @CsvFileSource:表示读取指定CSV文件内容作为参数化测试入参


  • @MethodSource:表示读取指定方法的返回值作为参数化测试入参(注意方法返回需要是一个流)


parameterized


 @ParameterizedTest
    @DisplayName("参数化测试")
    @ValueSource(ints = {1,2,3,4,5})
    void  testParameterized(int i){
        System.out.println(i);
    }
    //五个参数同时传进来的效果  把所有要测的统一测一遍
@ParameterizedTest
    @DisplayName("参数化测试")
    @MethodSource("stringProvider")
    void  testParameterized2(String i){
        System.out.println(i);
    }
    static Stream<String> stringProvider() {
        return Stream.of("apple", "banana","我最喜欢的大樱桃");
    }

junit4中的一些不可用了当你在junit4迁移过来的时候记得要看啊!


https://junit.org/junit5/docs/current/user-guide/#migrating-from-junit4


  • Annotations reside in the org.junit.jupiter.api package.


  • Assertions reside in org.junit.jupiter.api.Assertions.


  • Note that you may continue to use assertion methods from org.junit.Assert or any other assertion library such as AssertJ, Hamcrest, Truth, etc.
  • Assumptions reside in org.junit.jupiter.api.Assumptions.


  • Note that JUnit Jupiter 5.4 and later versions support methods from JUnit 4’s org.junit.Assume class for assumptions. Specifically, JUnit Jupiter supports JUnit 4’s AssumptionViolatedException to signal that a test should be aborted instead of marked as a failure.
  • @Before and @After no longer exist; use @BeforeEach and @AfterEach instead.


  • @BeforeClass and @AfterClass no longer exist; use @BeforeAll and @AfterAll instead.


  • @Ignore no longer exists: use @Disabled or one of the other built-in execution conditions instead


  • @Category no longer exists; use @Tag instead.


  • @RunWith no longer exists; superseded by @ExtendWith.


  • @Rule and @ClassRule no longer exist; superseded by @ExtendWith and @RegisterExtension


stead.


  • @BeforeClass and @AfterClass no longer exist; use @BeforeAll and @AfterAll instead.


  • @Ignore no longer exists: use @Disabled or one of the other built-in execution conditions instead


  • @Category no longer exists; use @Tag instead.


  • @RunWith no longer exists; superseded by @ExtendWith.


  • @Rule and @ClassRule no longer exist; superseded by @ExtendWith and @RegisterExtension


相关文章
|
5月前
|
Java Spring Apache
Spring Boot邂逅Apache Wicket:一次意想不到的完美邂逅,竟让Web开发变得如此简单?
【8月更文挑战第31天】Apache Wicket与Spring Boot的集成提供了近乎无缝的开发体验。Wicket以其简洁的API和强大的组件化设计著称,而Spring Boot则以开箱即用的便捷性赢得开发者青睐。本文将指导你如何在Spring Boot项目中引入Wicket,通过简单的步骤完成集成配置。首先,创建一个新的Spring Boot项目并在`pom.xml`中添加Wicket相关依赖。
138 0
|
7月前
|
安全 前端开发 Java
杨校老师课堂之Spring Boot框架面试题【开发工程师面试前必看】
杨校老师课堂之Spring Boot框架面试题【开发工程师面试前必看】
51 0
|
Java 测试技术 数据库连接
JUnit4教程+实践
JUnit是Java编程语言的单元测试框架,用于编写和可重复运行的自动化测试。
302 0
|
消息中间件 监控 前端开发
最新《阿里巴巴Java Spring Boot 2.0开发实战课程》持续更新 完全免费
最新《阿里巴巴Java Spring Boot 2.0开发实战课程》持续更新 完全免费第01课:Spring Boot2.0新特性和入门实战,https://yq.aliyun.com/live/583 第02课:Spring Boot2.
|
测试技术 Java
|
IDE Java 测试技术
JUnit 5 简介
著名的Java单元测试框架Junit 4已经出来很长时间了,当时我发现JUnit 5已经处于测试版,就准备写文章来介绍JUnit 5.不过因为还是测试版,所以有些地方还不太完善,我也有点懒没有好好写。
1214 0
|
测试技术 Apache Java
《JUnit实战(第2版)》—— 导读
作为一个屡获殊荣的数学家,我无法容忍平庸。这就是数学所教给我的─不要停止,直到你把它完成,并且不仅要用好的方法,而且要用最好的方法。
2547 0