JUnit5學習之七:參數化測試(Parameterized Tests)進階


歡迎訪問我的GitHub

https://github.com/zq2599/blog_demos

內容:所有原創文章分類匯總及配套源碼,涉及Java、Docker、Kubernetes、DevOPS等;

關於《JUnit5學習》系列

《JUnit5學習》系列旨在通過實戰提升SpringBoot環境下的單元測試技能,一共八篇文章,鏈接如下:

  1. 基本操作
  2. Assumptions類
  3. Assertions類
  4. 按條件執行
  5. 標簽(Tag)和自定義注解
  6. 參數化測試(Parameterized Tests)基礎
  7. 參數化測試(Parameterized Tests)進階
  8. 綜合進階(終篇)

本篇概覽

  • 本文是《JUnit5學習》系列的第七篇,前文咱們對JUnit5的參數化測試(Parameterized Tests)有了基本了解,可以使用各種數據源控制測試方法多次執行,今天要在此基礎上更加深入,掌握參數化測試的一些高級功能,解決實際問題;
  • 本文由以下章節組成:
  1. 自定義數據源
  2. 參數轉換
  3. 多字段聚合
  4. 多字段轉對象
  5. 測試執行名稱自定義

源碼下載

  1. 如果您不想編碼,可以在GitHub下載所有源碼,地址和鏈接信息如下表所示:
名稱 鏈接 備注
項目主頁 https://github.com/zq2599/blog_demos 該項目在GitHub上的主頁
git倉庫地址(https) https://github.com/zq2599/blog_demos.git 該項目源碼的倉庫地址,https協議
git倉庫地址(ssh) git@github.com:zq2599/blog_demos.git 該項目源碼的倉庫地址,ssh協議
  1. 這個git項目中有多個文件夾,本章的應用在junitpractice文件夾下,如下圖紅框所示:

在這里插入圖片描述

  1. junitpractice是父子結構的工程,本篇的代碼在parameterized子工程中,如下圖:

在這里插入圖片描述

自定義數據源

  1. 前文使用了很多種數據源,如果您對它們的各種限制不滿意,想要做更徹底的個性化定制,可以開發ArgumentsProvider接口的實現類,並使用@ArgumentsSource指定;
  2. 舉個例子,先開發ArgumentsProvider的實現類MyArgumentsProvider.java
package com.bolingcavalry.parameterized.service.impl;

import org.junit.jupiter.api.extension.ExtensionContext;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.ArgumentsProvider;
import java.util.stream.Stream;

public class MyArgumentsProvider implements ArgumentsProvider {

    @Override
    public Stream<? extends Arguments> provideArguments(ExtensionContext context) throws Exception {
        return Stream.of("apple4", "banana4").map(Arguments::of);
    }
}
  1. 再給測試方法添加@ArgumentsSource,並指定MyArgumentsProvider
    @Order(15)
    @DisplayName("ArgumentsProvider接口的實現類提供的數據作為入參")
    @ParameterizedTest
    @ArgumentsSource(MyArgumentsProvider.class)
    void argumentsSourceTest(String candidate) {
        log.info("argumentsSourceTest [{}]", candidate);
    }
  1. 執行結果如下:

在這里插入圖片描述

參數轉換

  1. 參數化測試的數據源和測試方法入參的數據類型必須要保持一致嗎?其實JUnit5並沒有嚴格要求,而事實上JUnit5是可以做一些自動或手動的類型轉換的;
  2. 如下代碼,數據源是int型數組,但測試方法的入參卻是double:
    @Order(16)
    @DisplayName("int型自動轉為double型入參")
    @ParameterizedTest
    @ValueSource(ints = { 1,2,3 })
    void argumentConversionTest(double candidate) {
        log.info("argumentConversionTest [{}]", candidate);
    }
  1. 執行結果如下,可見int型被轉為double型傳給測試方法(Widening Conversion):

在這里插入圖片描述

  1. 還可以指定轉換器,以轉換器的邏輯進行轉換,下面這個例子就是將字符串轉為LocalDate類型,關鍵是@JavaTimeConversionPattern
    @Order(17)
    @DisplayName("string型,指定轉換器,轉為LocalDate型入參")
    @ParameterizedTest
    @ValueSource(strings = { "01.01.2017", "31.12.2017" })
    void argumentConversionWithConverterTest(
            @JavaTimeConversionPattern("dd.MM.yyyy") LocalDate candidate) {
        log.info("argumentConversionWithConverterTest [{}]", candidate);
    }
  1. 執行結果如下:

在這里插入圖片描述

字段聚合(Argument Aggregation)

  1. 來思考一個問題:如果數據源的每條記錄有多個字段,測試方法如何才能使用這些字段呢?
  2. 回顧剛才的@CsvSource示例,如下圖,可見測試方法用兩個入參對應CSV每條記錄的兩個字段,如下所示:

在這里插入圖片描述
3. 上述方式應對少量字段還可以,但如果CSV每條記錄有很多字段,那測試方法豈不是要定義大量入參?這顯然不合適,此時可以考慮JUnit5提供的字段聚合功能(Argument Aggregation),也就是將CSV每條記錄的所有字段都放入一個ArgumentsAccessor類型的對象中,測試方法只要聲明ArgumentsAccessor類型作為入參,就能在方法內部取得CSV記錄的所有字段,效果如下圖,可見CSV字段實際上是保存在ArgumentsAccessor實例內部的一個Object數組中:

在這里插入圖片描述
4. 如下圖,為了方便從ArgumentsAccessor實例獲取數據,ArgumentsAccessor提供了獲取各種類型的方法,您可以按實際情況選用:

在這里插入圖片描述

  1. 下面的示例代碼中,CSV數據源的每條記錄有三個字段,而測試方法只有一個入參,類型是ArgumentsAccessor,在測試方法內部,可以用ArgumentsAccessor的getString、get等方法獲取CSV記錄的不同字段,例如arguments.getString(0)就是獲取第一個字段,得到的結果是字符串類型,而arguments.get(2, Types.class)的意思是獲取第二個字段,並且轉成了Type.class類型:
    @Order(18)
    @DisplayName("CsvSource的多個字段聚合到ArgumentsAccessor實例")
    @ParameterizedTest
    @CsvSource({
            "Jane1, Doe1, BIG",
            "John1, Doe1, SMALL"
    })
    void argumentsAccessorTest(ArgumentsAccessor arguments) {
        Person person = new Person();
        person.setFirstName(arguments.getString(0));
        person.setLastName(arguments.getString(1));
        person.setType(arguments.get(2, Types.class));

        log.info("argumentsAccessorTest [{}]", person);
    }
  1. 上述代碼執行結果如下圖,可見通過ArgumentsAccessor能夠取得CSV數據的所有字段:

在這里插入圖片描述

更優雅的聚合

  1. 前面的聚合解決了獲取CSV數據多個字段的問題,但依然有瑕疵:從ArgumentsAccessor獲取數據生成Person實例的代碼寫在了測試方法中,如下圖紅框所示,測試方法中應該只有單元測試的邏輯,而創建Person實例的代碼放在這里顯然並不合適:

在這里插入圖片描述
2. 針對上面的問題,JUnit5也給出了方案:通過注解的方式,指定一個從ArgumentsAccessor到Person的轉換器,示例如下,可見測試方法的入參有個注解@AggregateWith,其值PersonAggregator.class就是從ArgumentsAccessor到Person的轉換器,而入參已經從前面的ArgumentsAccessor變成了Person

    @Order(19)
    @DisplayName("CsvSource的多個字段,通過指定聚合類轉為Person實例")
    @ParameterizedTest
    @CsvSource({
            "Jane2, Doe2, SMALL",
            "John2, Doe2, UNKNOWN"
    })
    void customAggregatorTest(@AggregateWith(PersonAggregator.class) Person person) {
        log.info("customAggregatorTest [{}]", person);
    }
  1. PersonAggregator是轉換器類,需要實現ArgumentsAggregator接口,具體的實現代碼很簡單,也就是從ArgumentsAccessor示例獲取字段創建Person對象的操作:
package com.bolingcavalry.parameterized.service.impl;

import org.junit.jupiter.api.extension.ParameterContext;
import org.junit.jupiter.params.aggregator.ArgumentsAccessor;
import org.junit.jupiter.params.aggregator.ArgumentsAggregationException;
import org.junit.jupiter.params.aggregator.ArgumentsAggregator;

public class PersonAggregator implements ArgumentsAggregator {

    @Override
    public Object aggregateArguments(ArgumentsAccessor arguments, ParameterContext context) throws ArgumentsAggregationException {

        Person person = new Person();
        person.setFirstName(arguments.getString(0));
        person.setLastName(arguments.getString(1));
        person.setType(arguments.get(2, Types.class));

        return person;
    }
}
  1. 上述測試方法的執行結果如下:

在這里插入圖片描述

進一步簡化

  1. 回顧一下剛才用注解指定轉換器的代碼,如下圖紅框所示,您是否回憶起JUnit5支持自定義注解這一茬,咱們來把紅框部分的代碼再簡化一下:

在這里插入圖片描述
2. 新建注解類CsvToPerson.java,代碼如下,非常簡單,就是把上圖紅框中的@AggregateWith(PersonAggregator.class)搬過來了:

package com.bolingcavalry.parameterized.service.impl;

import org.junit.jupiter.params.aggregator.AggregateWith;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.PARAMETER)
@AggregateWith(PersonAggregator.class)
public @interface CsvToPerson {
}
  1. 再來看看上圖紅框中的代碼可以簡化成什么樣子,直接用@CsvToPerson就可以將ArgumentsAccessor轉為Person對象了:
    @Order(20)
    @DisplayName("CsvSource的多個字段,通過指定聚合類轉為Person實例(自定義注解)")
    @ParameterizedTest
    @CsvSource({
            "Jane3, Doe3, BIG",
            "John3, Doe3, UNKNOWN"
    })
    void customAggregatorAnnotationTest(@CsvToPerson Person person) {
        log.info("customAggregatorAnnotationTest [{}]", person);
    }
  1. 執行結果如下,可見和@AggregateWith(PersonAggregator.class)效果一致:

在這里插入圖片描述

測試執行名稱自定義

  1. 文章最后,咱們來看個輕松的知識點吧,如下圖紅框所示,每次執行測試方法,IDEA都會展示這次執行的序號和參數值:

在這里插入圖片描述

  1. 其實上述紅框中的內容格式也可以定制,格式模板就是@ParameterizedTestname屬性,修改后的測試方法完整代碼如下,可見這里改成了中文描述信息:
    @Order(21)
    @DisplayName("CSV格式多條記錄入參(自定義展示名稱)")
    @ParameterizedTest(name = "序號 [{index}],fruit參數 [{0}],rank參數 [{1}]")
    @CsvSource({
            "apple3, 31",
            "banana3, 32",
            "'lemon3, lime3', 0x3A"
    })
    void csvSourceWithCustomDisplayNameTest(String fruit, int rank) {
        log.info("csvSourceWithCustomDisplayNameTest, fruit [{}], rank [{}]", fruit, rank);
    }
  1. 執行結果如下:

在這里插入圖片描述

  • 至此,JUnit5的參數化測試(Parameterized)相關的知識點已經學習和實戰完成了,掌握了這么強大的參數輸入技術,咱們的單元測試的代碼覆蓋率和場景范圍又可以進一步提升了;

你不孤單,欣宸原創一路相伴

  1. Java系列
  2. Spring系列
  3. Docker系列
  4. kubernetes系列
  5. 數據庫+中間件系列
  6. DevOps系列

歡迎關注公眾號:程序員欣宸

微信搜索「程序員欣宸」,我是欣宸,期待與您一同暢游Java世界...
https://github.com/zq2599/blog_demos


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM