JUnit 5簡明教程

来源:https://www.cnblogs.com/okokabcd/archive/2022/07/13/16475909.html
-Advertisement-
Play Games

目前項目的目錄結構 簡要說明 因為項目是要分兩塊部署在不同的伺服器上的 task_producer是作為任務發佈者部署 task_customer是作為任務消費者 兩塊之前的消息通信的話,目前選用的是redis的隊列來進行的 消費者目前的代碼還沒有開始就暫時不展示了,下麵僅說明發佈者的目錄結構 cl ...


概述

寫測試用例對於開發來說有2點好處,一是開發階段寫完的功能可以快速驗證,第二就是在後期需求變動或修改BUG後可以快速測試當前改動是否帶來其它問題。下麵就瞭解一下Junit5寫測試用例。

準備

創建一個maven項目

mkdir junit5-tutorial
cd junit5-tutorial

mkdir -p src/main/java
mkdir -p src/test/java

mkdir -p src/main/resources
mkdir -p src/test/resources

# 編寫pom.xml
vi pom.xml

添加依賴

  • 引入第三方斷言庫assertj
  • 支持json測試
  • 支持xml測試

pom.xml

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example.xxx</groupId>
    <artifactId>junit5-tutorial</artifactId>
    <version>1.0-SNAPSHOT</version>

    <name>junit5-tutorial</name>

    <url>https://www.xxx.com</url>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>17</maven.compiler.source>
        <maven.compiler.target>17</maven.compiler.target>
    </properties>

    <build>
        <pluginManagement>
            <plugins>
                <!-- 在這裡聲明的目的是使用指定的版本 -->
                <!-- 執行測試用例任務的插件,預設綁定test生命周期的test階段 -->
                <plugin>
                    <artifactId>maven-surefire-plugin</artifactId>
                    <version>3.0.0-M6</version>
                </plugin>
                <!-- 用來執行編譯任務的插件,預設綁定default生命周期compile階段 -->
                <plugin>
                    <artifactId>maven-compiler-plugin</artifactId>
                    <version>3.10.1</version>
                </plugin>
            </plugins>
        </pluginManagement>

    </build>

    <dependencyManagement>
        <dependencies>
            <!-- XML Unit - Dependency Management -->
            <dependency>
                <groupId>net.bytebuddy</groupId>
                <artifactId>byte-buddy</artifactId>
                <version>1.12.10</version>
            </dependency>
            <dependency>
                <groupId>net.bytebuddy</groupId>
                <artifactId>byte-buddy-agent</artifactId>
                <version>1.12.10</version>
                <scope>test</scope>
            </dependency>

            <!-- Mockito Dependency -->
            <dependency>
                <groupId>org.mockito</groupId>
                <artifactId>mockito-junit-jupiter</artifactId>
                <version>4.5.1</version>
                <scope>test</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <dependencies>

        <dependency>
            <groupId>org.assertj</groupId>
            <artifactId>assertj-core</artifactId>
            <version>3.22.0</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>io.github.classgraph</groupId>
            <artifactId>classgraph</artifactId>
            <version>4.8.146</version>
        </dependency>
        <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter</artifactId>
            <version>5.8.2</version>
        </dependency>

        <!-- JSON Unit - Dependencies -->
        <dependency>
            <groupId>net.javacrumbs.json-unit</groupId>
            <artifactId>json-unit-assertj</artifactId>
            <version>2.33.0</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.13.2.2</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.datatype</groupId>
            <artifactId>jackson-datatype-jsr310</artifactId>
            <version>2.13.2</version>
        </dependency>

        <!-- XML Unit - Dependencies -->
        <dependency>
            <groupId>org.xmlunit</groupId>
            <artifactId>xmlunit-assertj</artifactId>
            <version>2.9.0</version>
            <scope>test</scope>
        </dependency>

    </dependencies>
</project>

創建一個User

public record User(String name, Integer age, Boolean blocked, LocalDate birthDate) {
}

測試

測試用例命名最佳實踐

首先測試類名應該以Test結尾,測試用例名稱最好遵從以下規則

  1. 測試名稱應表達特定要求
  2. 測試名稱應包含預期的輸入或預期的結果
  3. 測試名稱應以陳述的形式

具體參考:https://osherove.com/blog/2005/4/3/naming-standards-for-unit-tests.html

斷言

@Test
@DisplayName("User should be at least 18")
void user_should_be_at_least_18() {
    // junit5 的斷言
    assertTrue(user.age() >= 18);
    // assertj 的斷言
    assertThat(user.age()).isGreaterThanOrEqualTo(18);
}

顯示名稱

測試類和測試方法可以聲明自定義顯示名稱,可以使用空格、特殊字元、甚至emojis表情符號,這些名稱會在runner和測試報告中顯示。

參數化測試

參數化測試可以用不同的參數多次運行測試。它們和普通的@Test方法一樣聲明,但是使用@ParameterizedTest註解。還必須聲明至少一個將為每次調用提供參數的來

使用@ValueSource來指定參數來源

它可以指定一個原生類型的數組,並且只能為每次調用提供一個參數

@ParameterizedTest
@ValueSource(ints = {20, 50, 80})
void test_value_source(int age) {
    assertThat(age).isGreaterThanOrEqualTo(18);
}


讀取CSV文件內容作為參數來源

它可以讓你使用classpath中的csv文件。csv文件中的每一行都會導致參數測試的一次調用

src/test/resources/friends.csv

name,age
lisa,20
hans,30
hanna,40
@ParameterizedTest
@CsvFileSource(resources = "/friends.csv", numLinesToSkip = 1)
void test_value_source_by_csv_file_source(String name, int age) {
    assertThat(age).isGreaterThanOrEqualTo(18);
}

標簽

我們可以給測試類或測試用例上面通過@Tag加標簽,執行測試的時候可以指定標簽,從而達到為測試用例分組的目的。
下麵就給測試類打上一個integration的標簽

@Tag("integration")
class User01Test {
    // ...
}

可以使用如下命令來指定要執行的測試用例:

mvn test -Dgroups="integration"

左側只執行了Running com.example.xxx.User01Test一個測試類,右側則執行了3個

總結

本文介紹瞭如何使用Junit5寫測試用例。

參考

  1. Unit test naming best practices Unit test naming best practices - Stack Overflow
  2. https://osherove.com/blog/2005/4/3/naming-standards-for-unit-tests.html
  3. AssertJ Homepage https://assertj.github.io/doc/
  4. Gradle: https://stackoverflow.com/a/64986861
  5. https://junit.org/junit5/docs/current/user-guide/#running-tests-tag-expressions
  6. further reading
  7. JUnit 5 User Guide
  8. JUnit 5 中文文檔 https://doczhcn.gitbook.io/junit5/
  9. AssertJ - fluent assertions java library
  10. Jupiter / JUnit 5 - Testcontainers
  11. awaitility/awaitility: Awaitility is a small Java DSL for synchronizing asynchronous operations (github.com)

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在我們進行前後端聯調時,往往會出現後端給到的數據在前端不能直接做渲染的情況 這個時候就需要處理後端返回的數據,轉換數據結構前端再做渲染操作 當然如果和後端關係好~也可以讓後端改成你想要的樣子 以下簡單介紹遇到過的情況和相應的處理 覺得有用的話也可以自行封裝成工具類 後端返回線性結構數據,需根據rol ...
  • @import ​ Sass 拓展了 @import 的功能,允許其導入 SCSS 或 Sass 文件。被導入的文件將合併編譯到同一個 CSS 文件中,另外,被導入的文件中所包含的變數或者混合指令 (mixin) 都可以在導入的文件中使用。 例如: public.scss $font-base-co ...
  • 在平時,我非常喜歡利用 CSS 去構建一些有意思的圖形。 我們首先來看一個簡單的例子。首先,假設我們實現一個 10x10 的格子: 此時,我們可以利用一些隨機效果,優化這個圖案。譬如,我們給它隨機添加不同的顏色: 雖然利用了隨機,隨機填充了每一個格子的顏色,看著有那麼點意思,但是這隻是一幅雜亂無章的 ...
  • 簡單的封裝一下axios請求,包含了請求前的處理,返回結果和請求異常處理 import axios from "axios"; import store from "../store.js"; //封裝axios axios.defaults.headers["Content-Type"] = "a ...
  • 本文簡介 點贊 + 收藏 + 關註 = 學會了 這次會使用css畫出一個格子背景。並且一步步分析如何實現~ 思路 直接給答案:通過2個相等的直角三角形拼接,形成一個正方形。 三角形可以使用 background-image 的漸變來實現。 html, body { margin: 0; width: ...
  • 一、初識HTML 初始基本標簽代碼解釋 點擊查看代碼 <!--DOCTYPE:告訴瀏覽器我們要使用什麼規範--> <!DOCTYPE html> <html lang="en"> <!--生成註釋快捷鍵Ctrl + / 或 command + /--> <!--head標簽代表網頁頭部--> <he ...
  • python數據類型 進位轉換 變數 常量 運算符 ...
  • Java面向對象(五) 十六、面向對象特征之三: 多態性 16.1 多態性的定義: 對象的多態性:父類的引用指向子類的對象(或子類的對象賦給父類的引用)。 可以直接應用在抽象類和介面上。 Java 引用變數有兩個類型:編譯時類型和運行時類型。 編譯時類型由聲明該變數時使用的類型決定,運行時類型由實際 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...