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


相关文章
|
JavaScript Java 关系型数据库
毕设项目-基于Springboot和Vue实现蛋糕商城系统(一)
毕设项目-基于Springboot和Vue实现蛋糕商城系统
501 0
|
2月前
|
负载均衡 Java API
grpc-java 架构学习指南
本指南系统解析 grpc-java 架构,涵盖分层设计、核心流程与源码结构,结合实战路径与调试技巧,助你从入门到精通,掌握高性能 RPC 开发精髓。
256 7
|
安全 Java 数据安全/隐私保护
Stirling-PDF:一款优秀的开源PDF处理工具
Stirling-PDF是一个基于spring-boot开发的开源项目,旨在提供一个功能强大的基于Docker的本地托管PDF操作工具。它使您能够对PDF文件进行多种操作,包括拆分、合并、转换、重新组织、添加图片、旋转、压缩等。该本地托管应用最初由ChatGPT完全开发,并已发展成一个功能齐全的工具,可满足您的各种PDF需求。
6193 3
|
XML Java 数据格式
@Configuration配置类注解的理解
@Configuration配置类注解的理解
232 0
@Configuration配置类注解的理解
|
算法 调度 语音技术
======第三章处理机调度与死锁======(1)
第三章 处理机调度与死锁 3.1处理机调度的层次
112 0
|
设计模式 缓存 Java
好家伙!阿里新产Java性能优化(终极版),涵盖性能优化所有操作
上月公司来了一位大佬,入职不到一周就把公司现有项目的性能优化了一遍,直接给公司节省了一半的成本。 一问情况,才知道这位仁兄也是一路被虐过来的。去年年底被裁,本以为自己技术还行,看了一段时间面经,复习了基础知识,就开始投大厂简历。阿里最先给他面试机会,结果没能扛过三面,然后是各种大大小小的公司,在实际面试中被碾压得翻不了身。整整一个半月,一个offer都没拿到,最后针对性的恶补,才入职了我司。
|
JavaScript 前端开发 测试技术
Vue快速入门(附实战小项目:记事本、天气预报、音乐播放器)
Vue (读音 /vjuː/,类似于 view) 是一套用于构建用户界面的渐进式框架。 ​ Vue通过MVVM模式,能够实现视图与模型的双向绑定。 ​ 简单来说,就是数据变化的时候, 页面会自动刷新, 页面变化的时候,数据也会自动变化.
3769 0
Vue快速入门(附实战小项目:记事本、天气预报、音乐播放器)
|
消息中间件 Dubbo Java
【项目管理工具】maven快速入土
【项目管理工具】maven快速入土
294 0
【项目管理工具】maven快速入土
|
缓存 负载均衡 网络协议
一把王者的时间,我就学会了Nginx
Nginx("engine x") 是一个高性能的 HTTP 和反向代理服务器,特点是占有内存少,并发能力强,事实上 nginx 的并发能力确实在同类型的网页服务器中表现较好,中国大陆使用 nginx 网站用户有:百度、京东、新浪、网易、腾讯、 淘宝等。
199 0
一把王者的时间,我就学会了Nginx
|
算法 安全 Java
多线程初体验
程序:为完成特定任务,用某种编程语言编写的一组指令的结合(一段静态代码,静态对象) 进程:是程序的一次执行的过程或是正在运行的一个程序,是一个动态的过:有他自身的产生、存在、消亡的过程(生命周期),进程作为资源分配的单位,系统在运行时会为每个进程分配不同的内存区域。 线程:进程可以进一步细化为线程,是一个程序内部的一条执行路径。
183 0