一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - 詳解JUnit5參數化測試的幾種方式

詳解JUnit5參數化測試的幾種方式

2021-10-11 10:35測試開發客棧 Java教程

參數化測試一直是津津樂道的話題,我們都知道JMeter有四種參數化方式:用戶自定義變量、用戶參數、CSV文件、函數助手,那么JUnit5有哪些參數化測試的方式呢

參數化測試一直是津津樂道的話題,我們都知道JMeter有四種參數化方式:用戶自定義變量、用戶參數、CSV文件、函數助手,那么JUnit5有哪些參數化測試的方式呢?

依賴

JUnit5需要添加junit-jupiter-params依賴才能使用參數化:

?
1
2
3
4
5
6
<dependency>
    <groupId>org.junit.jupiter</groupId>
    <artifactId>junit-jupiter-params</artifactId>
    <version>5.7.2</version>
    <scope>test</scope>
</dependency>

簡單示例

@ParameterizedTest用來定義參數化測試,@ValueSource用來定義參數值:

?
1
2
3
4
5
@ParameterizedTest
@ValueSource(strings = { "racecar", "radar", "able was I ere I saw elba" })
void palindromes(String candidate) {
    assertTrue(StringUtils.isPalindrome(candidate));
}

執行結果:

?
1
2
3
4
palindromes(String) ?
├─ [1] candidate=racecar ?
├─ [2] candidate=radar ?
└─ [3] candidate=able was I ere I saw elba ?

參數值會匹配測試方法的參數列表,然后依次賦值,這里一共產生了3個測試。

七種方式

1 @ValueSource

@ValueSource是最簡單的參數化方式,它是一個數組,支持以下數據類型:

  • short
  • byte
  • int
  • long
  • float
  • double
  • char
  • boolean
  • java.lang.String
  • java.lang.Class

示例:

?
1
2
3
4
5
@ParameterizedTest
@ValueSource(ints = { 1, 2, 3 })
void testWithValueSource(int argument) {
    assertTrue(argument > 0 && argument < 4);
}

2 Null and Empty Sources

@NullSource 值為null

不能用在基元類型的測試方法。

@EmptySource 值為空,根據測試方法的參數類決定數據類型,支持java.lang.String, java.util.List, java.util.Set, java.util.Map, 基元類型數組 (int[], char[][]等), 對象數組 (String[], Integer[][]等)

@NullAndEmptySource 結合了前面兩個

示例:

?
1
2
3
4
5
6
7
@ParameterizedTest
@NullSource
@EmptySource
@ValueSource(strings = { " ", "   ", "\t", "\n" })
void nullEmptyAndBlankStrings(String text) {
    assertTrue(text == null || text.trim().isEmpty());
}

等價于:

?
1
2
3
4
5
6
@ParameterizedTest
@NullAndEmptySource
@ValueSource(strings = { " ", "   ", "\t", "\n" })
void nullEmptyAndBlankStrings(String text) {
    assertTrue(text == null || text.trim().isEmpty());
}

3 @EnumSource

參數化的值為枚舉類型。

示例:

?
1
2
3
4
5
@ParameterizedTest
@EnumSource
void testWithEnumSourceWithAutoDetection(ChronoUnit unit) {
    assertNotNull(unit);
}

其中的ChronoUnit是個日期枚舉類。

ChronoUnit是接口TemporalUnit的實現類,如果測試方法的參數為TemporalUnit,那么需要給@EnumSource加上值:

?
1
2
3
4
5
@ParameterizedTest
@EnumSource(ChronoUnit.class)
void testWithEnumSource(TemporalUnit unit) {
    assertNotNull(unit);
}

因為JUnit5規定了@EnumSource的默認值的類型必須是枚舉類型。

names屬性用來指定使用哪些特定的枚舉值:

?
1
2
3
4
5
@ParameterizedTest
@EnumSource(names = { "DAYS", "HOURS" })
void testWithEnumSourceInclude(ChronoUnit unit) {
    assertTrue(EnumSet.of(ChronoUnit.DAYS, ChronoUnit.HOURS).contains(unit));
}

mode屬性用來指定使用模式,比如排除哪些枚舉值:

?
1
2
3
4
5
@ParameterizedTest
@EnumSource(mode = EXCLUDE, names = { "ERAS", "FOREVER" })
void testWithEnumSourceExclude(ChronoUnit unit) {
    assertFalse(EnumSet.of(ChronoUnit.ERAS, ChronoUnit.FOREVER).contains(unit));
}

比如采用正則匹配:

?
1
2
3
4
5
@ParameterizedTest
@EnumSource(mode = MATCH_ALL, names = "^.*DAYS$")
void testWithEnumSourceRegex(ChronoUnit unit) {
    assertTrue(unit.name().endsWith("DAYS"));
}

4 @MethodSource

參數值為factory方法,并且factory方法不能帶參數。

示例:

?
1
2
3
4
5
6
7
8
9
@ParameterizedTest
@MethodSource("stringProvider")
void testWithExplicitLocalMethodSource(String argument) {
    assertNotNull(argument);
}
 
static Stream<String> stringProvider() {
    return Stream.of("apple", "banana");
}

除非是@TestInstance(Lifecycle.PER_CLASS)生命周期,否則factory方法必須是static。factory方法的返回值是能轉換為Stream的類型,比如Stream, DoubleStream, LongStream, IntStream, Collection, Iterator, Iterable, 對象數組, 或者基元類型數組,比如:

?
1
2
3
4
5
6
7
8
9
@ParameterizedTest
@MethodSource("range")
void testWithRangeMethodSource(int argument) {
    assertNotEquals(9, argument);
}
 
static IntStream range() {
    return IntStream.range(0, 20).skip(10);
}

@MethodSource的屬性如果省略了,那么JUnit Jupiter會找跟測試方法同名的factory方法,比如:

?
1
2
3
4
5
6
7
8
9
@ParameterizedTest
@MethodSource
void testWithDefaultLocalMethodSource(String argument) {
    assertNotNull(argument);
}
 
static Stream<String> testWithDefaultLocalMethodSource() {
    return Stream.of("apple", "banana");
}

如果測試方法有多個參數,那么factory方法也應該返回多個:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@ParameterizedTest
@MethodSource("stringIntAndListProvider")
void testWithMultiArgMethodSource(String str, int num, List<String> list) {
    assertEquals(5, str.length());
    assertTrue(num >=1 && num <=2);
    assertEquals(2, list.size());
}
 
static Stream<Arguments> stringIntAndListProvider() {
    return Stream.of(
        arguments("apple", 1, Arrays.asList("a", "b")),
        arguments("lemon", 2, Arrays.asList("x", "y"))
    );
}

其中arguments(Object…)是Arguments接口的static factory method,也可以換成Arguments.of(Object…)。

factory方法也可以防止測試類外部:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
package example;
 
import java.util.stream.Stream;
 
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.MethodSource;
 
class ExternalMethodSourceDemo {
 
    @ParameterizedTest
    @MethodSource("example.StringsProviders#tinyStrings")
    void testWithExternalMethodSource(String tinyString) {
        // test with tiny string
    }
}
 
class StringsProviders {
 
    static Stream<String> tinyStrings() {
        return Stream.of(".", "oo", "OOO");
    }
}

5 @CsvSource

參數化的值為csv格式的數據(默認逗號分隔),比如:

?
1
2
3
4
5
6
7
8
9
10
@ParameterizedTest
@CsvSource({
    "apple,         1",
    "banana,        2",
    "'lemon, lime', 0xF1"
})
void testWithCsvSource(String fruit, int rank) {
    assertNotNull(fruit);
    assertNotEquals(0, rank);
}

delimiter屬性可以設置分隔字符。delimiterString屬性可以設置分隔字符串(String而非char)。

更多輸入輸出示例如下:

詳解JUnit5參數化測試的幾種方式

注意,如果null引用的目標類型是基元類型,那么會報異常ArgumentConversionException。

6 @CsvFileSource

顧名思義,選擇本地csv文件作為數據來源。

示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
@ParameterizedTest
@CsvFileSource(resources = "/two-column.csv", numLinesToSkip = 1)
void testWithCsvFileSourceFromClasspath(String country, int reference) {
    assertNotNull(country);
    assertNotEquals(0, reference);
}
 
@ParameterizedTest
@CsvFileSource(files = "src/test/resources/two-column.csv", numLinesToSkip = 1)
void testWithCsvFileSourceFromFile(String country, int reference) {
    assertNotNull(country);
    assertNotEquals(0, reference);
}

delimiter屬性可以設置分隔字符。delimiterString屬性可以設置分隔字符串(String而非char)。需要特別注意的是,#開頭的行會被認為是注釋而略過。

7 @ArgumentsSource

自定義ArgumentsProvider。

示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
@ParameterizedTest
@ArgumentsSource(MyArgumentsProvider.class)
void testWithArgumentsSource(String argument) {
    assertNotNull(argument);
}
public class MyArgumentsProvider implements ArgumentsProvider {
 
    @Override
    public Stream<? extends Arguments> provideArguments(ExtensionContext context) {
        return Stream.of("apple", "banana").map(Arguments::of);
    }
}

MyArgumentsProvider必須是外部類或者static內部類。

參數類型轉換

隱式轉換

JUnit Jupiter會對String類型進行隱式轉換。比如:

?
1
2
3
4
5
@ParameterizedTest
@ValueSource(strings = "SECONDS")
void testWithImplicitArgumentConversion(ChronoUnit argument) {
    assertNotNull(argument.name());
}

更多轉換示例:

詳解JUnit5參數化測試的幾種方式

詳解JUnit5參數化測試的幾種方式

也可以把String轉換為自定義對象:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@ParameterizedTest
@ValueSource(strings = "42 Cats")
void testWithImplicitFallbackArgumentConversion(Book book) {
    assertEquals("42 Cats", book.getTitle());
}
public class Book {
 
    private final String title;
 
    private Book(String title) {
        this.title = title;
    }
 
    public static Book fromTitle(String title) {
        return new Book(title);
    }
 
    public String getTitle() {
        return this.title;
    }
}

JUnit Jupiter會找到Book.fromTitle(String)方法,然后把@ValueSource的值傳入進去,進而把String類型轉換為Book類型。轉換的factory方法既可以是接受單個String參數的構造方法,也可以是接受單個String參數并返回目標類型的普通方法。詳細規則如下(官方原文):

詳解JUnit5參數化測試的幾種方式

顯式轉換

顯式轉換需要使用@ConvertWith注解:

?
1
2
3
4
5
6
7
@ParameterizedTest
@EnumSource(ChronoUnit.class)
void testWithExplicitArgumentConversion(
        @ConvertWith(ToStringArgumentConverter.class) String argument) {
 
    assertNotNull(ChronoUnit.valueOf(argument));
}

并實現ArgumentConverter:

?
1
2
3
4
5
6
7
8
9
10
11
public class ToStringArgumentConverter extends SimpleArgumentConverter {
 
    @Override
    protected Object convert(Object source, Class<?> targetType) {
        assertEquals(String.class, targetType, "Can only convert to String");
        if (source instanceof Enum<?>) {
            return ((Enum<?>) source).name();
        }
        return String.valueOf(source);
    }
}

如果只是簡單類型轉換,實現TypedArgumentConverter即可:

?
1
2
3
4
5
6
7
8
9
10
11
12
public class ToLengthArgumentConverter extends TypedArgumentConverter<String, Integer> {
 
    protected ToLengthArgumentConverter() {
        super(String.class, Integer.class);
    }
 
    @Override
    protected Integer convert(String source) {
        return source.length();
    }
 
}

JUnit Jupiter只內置了一個JavaTimeArgumentConverter,通過@JavaTimeConversionPattern使用:

?
1
2
3
4
5
6
7
@ParameterizedTest
@ValueSource(strings = { "01.01.2017", "31.12.2017" })
void testWithExplicitJavaTimeConverter(
        @JavaTimeConversionPattern("dd.MM.yyyy") LocalDate argument) {
 
    assertEquals(2017, argument.getYear());
}

參數聚合

測試方法的多個參數可以聚合為一個ArgumentsAccessor參數,然后通過get來取值,示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@ParameterizedTest
@CsvSource({
    "Jane, Doe, F, 1990-05-20",
    "John, Doe, M, 1990-10-22"
})
void testWithArgumentsAccessor(ArgumentsAccessor arguments) {
    Person person = new Person(arguments.getString(0),
                               arguments.getString(1),
                               arguments.get(2, Gender.class),
                               arguments.get(3, LocalDate.class));
 
    if (person.getFirstName().equals("Jane")) {
        assertEquals(Gender.F, person.getGender());
    }
    else {
        assertEquals(Gender.M, person.getGender());
    }
    assertEquals("Doe", person.getLastName());
    assertEquals(1990, person.getDateOfBirth().getYear());
}

也可以自定義Aggregator:

?
1
2
3
4
5
6
7
8
9
public class PersonAggregator implements ArgumentsAggregator {
    @Override
    public Person aggregateArguments(ArgumentsAccessor arguments, ParameterContext context) {
        return new Person(arguments.getString(0),
                          arguments.getString(1),
                          arguments.get(2, Gender.class),
                          arguments.get(3, LocalDate.class));
    }
}

然后通過@AggregateWith來使用:

?
1
2
3
4
5
6
7
8
@ParameterizedTest
@CsvSource({
    "Jane, Doe, F, 1990-05-20",
    "John, Doe, M, 1990-10-22"
})
void testWithArgumentsAggregator(@AggregateWith(PersonAggregator.class) Person person) {
    // perform assertions against person
}

借助于組合注解,我們可以進一步簡化代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.PARAMETER)
@AggregateWith(PersonAggregator.class)
public @interface CsvToPerson {
}
@ParameterizedTest
@CsvSource({
    "Jane, Doe, F, 1990-05-20",
    "John, Doe, M, 1990-10-22"
})
void testWithCustomAggregatorAnnotation(@CsvToPerson Person person) {
    // perform assertions against person
}

自定義顯示名字

參數化測試生成的test,JUnit Jupiter給定了默認名字,我們可以通過name屬性進行自定義。

示例:

?
1
2
3
4
5
@DisplayName("Display name of container")
@ParameterizedTest(name = "{index} ==> the rank of ''{0}'' is {1}")
@CsvSource({ "apple, 1", "banana, 2", "'lemon, lime', 3" })
void testWithCustomDisplayNames(String fruit, int rank) {
}

結果:

?
1
2
3
4
Display name of container ?
├─ 1 ==> the rank of 'apple' is 1 ?
├─ 2 ==> the rank of 'banana' is 2 ?
└─ 3 ==> the rank of 'lemon, lime' is 3 ?

注意如果要顯示'apple',需要使用兩層''apple'',因為name是MessageFormat。

占位符說明如下:

詳解JUnit5參數化測試的幾種方式

小結

本文介紹了JUnit5參數化測試的7種方式,分別是@ValueSource,Null and Empty Sources,@EnumSource,@MethodSource@CsvSource@CsvFileSource,@ArgumentsSource,比較偏向于Java語法,符合JUnit單元測試框架的特征。另外還介紹了JUnit Jupiter的參數類型轉換和參數聚合。最后,如果想要自定義參數化測試的名字,可以使用name屬性實現。

參考資料:

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

到此這篇關于詳解JUnit5參數化測試的幾種方式的文章就介紹到這了,更多相關JUnit5參數化測試 內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://www.cnblogs.com/df888/p/15013017.html

延伸 · 閱讀

精彩推薦
  • Java教程20個非常實用的Java程序代碼片段

    20個非常實用的Java程序代碼片段

    這篇文章主要為大家分享了20個非常實用的Java程序片段,對java開發項目有所幫助,感興趣的小伙伴們可以參考一下 ...

    lijiao5352020-04-06
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    這篇文章主要介紹了Java使用SAX解析xml的示例,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程xml與Java對象的轉換詳解

    xml與Java對象的轉換詳解

    這篇文章主要介紹了xml與Java對象的轉換詳解的相關資料,需要的朋友可以參考下...

    Java教程網2942020-09-17
  • Java教程小米推送Java代碼

    小米推送Java代碼

    今天小編就為大家分享一篇關于小米推送Java代碼,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧...

    富貴穩中求8032021-07-12
  • Java教程升級IDEA后Lombok不能使用的解決方法

    升級IDEA后Lombok不能使用的解決方法

    最近看到提示IDEA提示升級,尋思已經有好久沒有升過級了。升級完畢重啟之后,突然發現好多錯誤,本文就來介紹一下如何解決,感興趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java8中Stream使用的一個注意事項

    Java8中Stream使用的一個注意事項

    最近在工作中發現了對于集合操作轉換的神器,java8新特性 stream,但在使用中遇到了一個非常重要的注意點,所以這篇文章主要給大家介紹了關于Java8中S...

    阿杜7472021-02-04
  • Java教程Java BufferWriter寫文件寫不進去或缺失數據的解決

    Java BufferWriter寫文件寫不進去或缺失數據的解決

    這篇文章主要介紹了Java BufferWriter寫文件寫不進去或缺失數據的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望...

    spcoder14552021-10-18
  • Java教程Java實現搶紅包功能

    Java實現搶紅包功能

    這篇文章主要為大家詳細介紹了Java實現搶紅包功能,采用多線程模擬多人同時搶紅包,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙...

    littleschemer13532021-05-16
主站蜘蛛池模板: 天天爱天天插 | 99草视频 | 国产欧美日韩精品一区二区三区 | 国产小嫩模好紧 | 99久久精品免费看国产 | 国产无限 | 日本免费高清在线 | 亚洲大爷操 | 我的男友是消防员在线观看 | 日本黄色录像视频 | 日本无吗免费一二区 | 近亲乱中文字幕 | 王小军怎么了最新消息 | 成人永久免费福利视频网站 | 国内精品一区二区三区东京 | 桃乃木香奈ipx在线播放 | 亚洲色域网 | 国产午夜精品福利久久 | 大学生宿舍飞机china free | 美女福利视频网站 | 扒开老师两片湿漉的肉 | 青青成人福利国产在线视频 | 国产欧美国产综合第一区 | 调教女高中生第3部分 | 操美女骚b | 欧美日韩亚洲国内综合网香蕉 | 女bbbbxxx孕妇 | 免费在线观看中文字幕 | jux629三浦理惠子在线播放 | 王晶经典三级 | 无限在线观看视频大全免费高清 | 999国产| 800精品国产导航 | 美女视频91 | 日韩一区二区在线视频 | 男女性刺激爽爽免费视频 | 亚洲无限| 果冻传媒在线免费观看 | 逼水真多 | 青草悠悠视频在线观看 | 女同xx美女放 |