Java單元測試之JUnit 5快速上手

原文:http://www.reibang.com/p/a2aaa020cb03

前言

單元測試是軟件開發(fā)中必不可少的一環(huán)状您,但是在平常開發(fā)中往往因為項目周期緊勒叠,工作量大而被選擇忽略,這樣往往導(dǎo)致軟件問題層出不窮膏孟。線上出現(xiàn)的不少問題其實在有單元測試的情況下就可以及時發(fā)現(xiàn)和處理眯分,因此培養(yǎng)自己在日常開發(fā)中寫單元測試的能力是很有必要的。無論是對自己的編碼能力的提高骆莹,還是項目質(zhì)量的提升颗搂,都是大有好處,本文將介紹 Java 單元測試框架 JUnit 5 的基礎(chǔ)認識和使用來編寫單元測試幕垦,希望同樣對你有所幫助。

本文所涉及所有代碼片段均在下面?zhèn)}庫中傅联,感興趣的小伙伴歡迎參考學習:

https://github.com/wrcj12138aaa/junit5-actions

版本支持:

  • JDK 8
  • JUnit 5.5.2
  • Lomok 1.18.8

認識 JUnit 5

要說什么是 JUnit 5先改,首先就得聊下 Java 單元測試框架 JUnit,它與另一個框架 TestNG 占據(jù)了 Java領(lǐng)域里單元測試框架的主要市場蒸走,其中 JUnit 有著較長的發(fā)展歷史和不斷演進的豐富功能仇奶,備受大多數(shù) Java 開發(fā)者的青睞。

而說到 JUnit 的歷史比驻,JUnit 起源于 1997年该溯,最初版本是由兩位編程大師 Kent Beck 和 Erich Gamma 的一次飛機之旅上完成的,由于當時 Java 測試過程中缺乏成熟的工具别惦,兩人在飛機上就合作設(shè)計實現(xiàn)了 JUnit 雛形狈茉,旨在成為更好用的 Java 測試框架。如今二十多年過去了掸掸,JUnit 經(jīng)過各個版本迭代演進氯庆,已經(jīng)發(fā)展到了 5.x 版本,為 JDK 8以及更高的版本上提供更好的支持 (如支持 Lambda ) 和更豐富的測試形式 (如重復(fù)測試扰付,參數(shù)化測試)堤撵。

了解過 JUint 之后,再回頭來看下 JUnit 5羽莺,這個版本可以說是 JUnit 單元測試框架的一次重大升級实昨,首先需要 Java 8 以上的運行環(huán)境,雖然在舊版本 JDK 也能編譯運行盐固,但要完全使用 JUnit 5 功能荒给, JDK 8 環(huán)境是必不可少的丈挟。

除此之外,JUnit 5 與以前版本的 JUnit 不同锐墙,拆分成由三個不同子項目的幾個不同模塊組成礁哄。

JUnit 5 = JUnit Platform + JUnit Jupiter + JUnit Vintage

  • JUnit Platform: 用于JVM上啟動測試框架的基礎(chǔ)服務(wù),提供命令行溪北,IDE和構(gòu)建工具等方式執(zhí)行測試的支持桐绒。

  • JUnit Jupiter:包含 JUnit 5 新的編程模型和擴展模型,主要就是用于編寫測試代碼和擴展代碼之拨。

  • JUnit Vintage:用于在JUnit 5 中兼容運行 JUnit3.x 和 JUnit4.x 的測試用例茉继。

基于上面的介紹,可以參考下圖對 JUnit 5 的架構(gòu)和模塊有所了解:

image

為什么需要 JUnit 5

說完 JUnit 5 是什么之后蚀乔,我們再來想一個問題:為什么需要一個 JUnit 5 呢烁竭?

自從有了類似 JUnit 之類的測試框架,Java 單元測試領(lǐng)域逐漸成熟吉挣,開發(fā)人員對單元測試框架也有了更高的要求:更多的測試方式派撕,更少的其他庫的依賴。因此睬魂,大家期待著一個更強大的測試框架誕生终吼,JUnit 作為Java測試領(lǐng)域的領(lǐng)頭羊,推出了 JUnit 5 這個版本氯哮,主要特性:

  • 提供全新的斷言和測試注解际跪,支持測試類內(nèi)嵌

  • 更豐富的測試方式:支持動態(tài)測試,重復(fù)測試喉钢,參數(shù)化測試等

  • 實現(xiàn)了模塊化姆打,讓測試執(zhí)行和測試發(fā)現(xiàn)等不同模塊解耦,減少依賴

  • 提供對 Java 8 的支持肠虽,如 Lambda 表達式幔戏,Sream API等。

JUnit 5 常見用法介紹

接下來舔痕,我們看下 JUni 5 的一些常見用法评抚,來幫助我們快速掌握 JUnit 5 的使用。

首先伯复,在 Maven 工程里引入 JUnit 5 的依賴坐標慨代,需注意的是當前JDK 環(huán)境要在 Java 8 以上。

<dependency>
  <groupId>org.junit.jupiter</groupId>
  <artifactId>junit-jupiter-engine</artifactId>
  <version>5.5.2</version>
  <scope>test</scope>
</dependency>

第一個測試用例

引入JUnit 5啸如,我們可以先快速編寫一個簡單的測試用例侍匙,從這個測試用例來認識初步下 JUnit 5:

@DisplayName("我的第一個測試用例")
public class MyFirstTestCaseTest {

    @BeforeAll
    public static void init() {
        System.out.println("初始化數(shù)據(jù)");
    }

    @AfterAll
    public static void cleanup() {
        System.out.println("清理數(shù)據(jù)");
    }

    @BeforeEach
    public void tearup() {
        System.out.println("當前測試方法開始");
    }

    @AfterEach
    public void tearDown() {
        System.out.println("當前測試方法結(jié)束");
    }

    @DisplayName("我的第一個測試")
    @Test
    void testFirstTest() {
        System.out.println("我的第一個測試開始測試");
    }

    @DisplayName("我的第二個測試")
    @Test
    void testSecondTest() {
        System.out.println("我的第二個測試開始測試");
    }
}

直接運行這個測試用例,可以看到控制臺日志如下:

image

可以看到左邊一欄的結(jié)果里顯示測試項名稱就是我們在測試類和方法上使用 @DisplayName 設(shè)置的名稱,這個注解就是 JUnit 5 引入想暗,用來定義一個測試類并指定用例在測試報告中的展示名稱妇汗,這個注解可以使用在類上和方法上,在類上使用它就表示該類為測試類说莫,在方法上使用則表示該方法為測試方法杨箭。

@Target({ ElementType.TYPE, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@API(status = STABLE, since = "5.0")
public @interface DisplayName {
    String value();
}

再來看下示例代碼中使用到的一對注解 **@BeforeAll **和 @AfterAll ,它們定義了整個測試類在開始前以及結(jié)束時的操作储狭,只能修飾靜態(tài)方法互婿,主要用于在測試過程中所需要的全局數(shù)據(jù)和外部資源的初始化和清理。與它們不同辽狈,@BeforeEach@AfterEach 所標注的方法會在每個測試用例方法開始前和結(jié)束時執(zhí)行慈参,主要是負責該測試用例所需要的運行環(huán)境的準備和銷毀。

在測試過程中除了這些基本的注解刮萌,還有更多豐富強大的注解驮配,接下來就我們一一學習下吧。

禁用執(zhí)行測試:@Disabled

當我們希望在運行測試類時着茸,跳過某個測試方法壮锻,正常運行其他測試用例時,我們就可以用上 @Disabled 注解涮阔,表明該測試方法處于不可用躯保,執(zhí)行測試類的測試方法時不會被 JUnit 執(zhí)行。

下面看下使用 @Disbaled 之后的運行效果澎语,在原來測試類中添加如下代碼:

@DisplayName("我的第三個測試")
@Disabled
@Test
void testThirdTest() {
    System.out.println("我的第三個測試開始測試");
}

運行后看到控制臺日志如下,用 @Disabled 標記的方法不會執(zhí)行验懊,只有單獨的方法信息打由眯摺:

image

@Disabled 也可以使用在類上,用于標記類下所有的測試方法不被執(zhí)行义图,一般使用對多個測試類組合測試的時候减俏。

內(nèi)嵌測試類:@Nested

當我們編寫的類和代碼逐漸增多,隨之而來的需要測試的對應(yīng)測試類也會越來越多碱工。為了解決測試類數(shù)量爆炸的問題娃承,JUnit 5提供了@Nested 注解,能夠以靜態(tài)內(nèi)部成員類的形式對測試用例類進行邏輯分組怕篷。 并且每個靜態(tài)內(nèi)部類都可以有自己的生命周期方法历筝, 這些方法將按從外到內(nèi)層次順序執(zhí)行。 此外廊谓,嵌套的類也可以用@DisplayName 標記梳猪,這樣我們就可以使用正確的測試名稱。下面看下簡單的用法:

@DisplayName("內(nèi)嵌測試類")
public class NestUnitTest {
    @BeforeEach
    void init() {
        System.out.println("測試方法執(zhí)行前準備");
    }

    @Nested
    @DisplayName("第一個內(nèi)嵌測試類")
    class FirstNestTest {
        @Test
        void test() {
            System.out.println("第一個內(nèi)嵌測試類執(zhí)行測試");
        }
    }

    @Nested
    @DisplayName("第二個內(nèi)嵌測試類")
    class SecondNestTest {
        @Test
        void test() {
            System.out.println("第二個內(nèi)嵌測試類執(zhí)行測試");
        }
    }
}

運行所有測試用例后蒸痹,在控制臺能看到如下結(jié)果:

image

重復(fù)性測試:@RepeatedTest

在 JUnit 5 里新增了對測試方法設(shè)置運行次數(shù)的支持春弥,允許讓測試方法進行重復(fù)運行呛哟。當要運行一個測試方法 N次時,可以使用 @RepeatedTest 標記它匿沛,如下面的代碼所示:

@DisplayName("重復(fù)測試")
@RepeatedTest(value = 3)
public void i_am_a_repeated_test() {
    System.out.println("執(zhí)行測試");
}

運行后測試方法會執(zhí)行3次扫责,在 IDEA 的運行效果如下圖所示:

image

這是基本的用法,我們還可以對重復(fù)運行的測試方法名稱進行修改逃呼,利用 @RepeatedTest 提供的內(nèi)置變量鳖孤,以占位符方式在其 name 屬性上使用,下面先看下使用方式和效果:

@DisplayName("自定義名稱重復(fù)測試")
@RepeatedTest(value = 3, name = "{displayName} 第 {currentRepetition} 次")
public void i_am_a_repeated_test_2() {
    System.out.println("執(zhí)行測試");
}

image

@RepeatedTest 注解內(nèi)用 currentRepetition 變量表示已經(jīng)重復(fù)的次數(shù)蜘渣,totalRepetitions 變量表示總共要重復(fù)的次數(shù)淌铐,displayName 變量表示測試方法顯示名稱,我們直接就可以使用這些內(nèi)置的變量來重新定義測試方法重復(fù)運行時的名稱蔫缸。

新的斷言

在斷言 API 設(shè)計上腿准,JUnit 5 進行顯著地改進,并且充分利用 Java 8 的新特性拾碌,特別是 Lambda 表達式吐葱,最終提供了新的斷言類: org.junit.jupiter.api.Assertions 。許多斷言方法接受 Lambda 表達式參數(shù)校翔,在斷言消息使用 Lambda 表達式的一個優(yōu)點就是它是延遲計算的弟跑,如果消息構(gòu)造開銷很大,這樣做一定程度上可以節(jié)省時間和資源防症。

現(xiàn)在還可以將一個方法內(nèi)的多個斷言進行分組孟辑,使用 assertAll 方法如下示例代碼:

@Test
void testGroupAssertions() {
    int[] numbers = {0, 1, 2, 3, 4};
    Assertions.assertAll("numbers",
            () -> Assertions.assertEquals(numbers[1], 1),
            () -> Assertions.assertEquals(numbers[3], 3),
            () -> Assertions.assertEquals(numbers[4], 4)
    );
}

如果分組斷言中任一個斷言的失敗,都會將以 MultipleFailuresError 錯誤進行拋出提示蔫敲。

超時操作的測試:assertTimeoutPreemptively

當我們希望測試耗時方法的執(zhí)行時間饲嗽,并不想讓測試方法無限地等待時,就可以對測試方法進行超時測試奈嘿,JUnit 5 對此推出了斷言方法 assertTimeout貌虾,提供了對超時的廣泛支持。

假設(shè)我們希望測試代碼在一秒內(nèi)執(zhí)行完畢裙犹,可以寫如下測試用例:

@Test
@DisplayName("超時方法測試")
void test_should_complete_in_one_second() {
  Assertions.assertTimeoutPreemptively(Duration.of(1, ChronoUnit.SECONDS), () -> Thread.sleep(2000));
}

這個測試運行失敗尽狠,因為代碼執(zhí)行將休眠兩秒鐘,而我們期望測試用例在一秒鐘之內(nèi)成功叶圃。但是如果我們把休眠時間設(shè)置一秒鐘袄膏,測試仍然會出現(xiàn)偶爾失敗的情況,這是因為測試方法執(zhí)行過程中除了目標代碼還有額外的代碼和指令執(zhí)行會耗時盗似,所以在超時限制上無法做到對時間參數(shù)的完全精確匹配哩陕。

異常測試:assertThrows

我們代碼中對于帶有異常的方法通常都是使用 try-catch 方式捕獲處理,針對測試這樣帶有異常拋出的代碼,而 JUnit 5 提供方法 Assertions#assertThrows(Class<T>, Executable) 來進行測試悍及,第一個參數(shù)為異常類型闽瓢,第二個為函數(shù)式接口參數(shù),跟 Runnable 接口相似心赶,不需要參數(shù)扣讼,也沒有返回,并且支持 Lambda表達式方式使用缨叫,具體使用方式可參考下方代碼:

@Test
@DisplayName("測試捕獲的異常")
void assertThrowsException() {
  String str = null;
  Assertions.assertThrows(IllegalArgumentException.class, () -> {
    Integer.valueOf(str);
  });
}

當Lambda表達式中代碼出現(xiàn)的異常會跟首個參數(shù)的異常類型進行比較椭符,如果不屬于同一類異常,就會控制臺輸出如下類似的提示:org.opentest4j.AssertionFailedError: Unexpected exception type thrown ==> expected: <IllegalArgumentException> but was: <...Exception>

JUnit 5 參數(shù)化測試

要使用 JUnit 5 進行參數(shù)化測試耻姥,除了 junit-jupiter-engine 基礎(chǔ)依賴之外销钝,還需要另個模塊依賴:junit-jupiter-params,其主要就是提供了編寫參數(shù)化測試 API琐簇。同樣方式蒸健,把相同版本的對應(yīng)依賴引入 Maven 工程中:

<dependency>
  <groupId>org.junit.jupiter</groupId>
  <artifactId>junit-jupiter-params</artifactId>
  <version>5.5.2</version>
  <scope>test</scope>
</dependency>

基本數(shù)據(jù)源測試: @ValueSource

@ValueSource 是 JUnit 5 提供的最簡單的數(shù)據(jù)參數(shù)源,支持 Java 的八大基本類型和字符串婉商,Class似忧,使用時賦值給注解上對應(yīng)類型屬性,以數(shù)組方式傳遞丈秩,示例代碼如下:

public class ParameterizedUnitTest {
    @ParameterizedTest
    @ValueSource(ints = {2, 4, 8})
    void testNumberShouldBeEven(int num) {
        Assertions.assertEquals(0, num % 2);
    }

    @ParameterizedTest
    @ValueSource(strings = {"Effective Java", "Code Complete", "Clean Code"})
    void testPrintTitle(String title) {
        System.out.println(title);
    }
}

@ParameterizedTest 作為參數(shù)化測試的必要注解盯捌,替代了 @Test 注解。任何一個參數(shù)化測試方法都需要標記上該注解蘑秽。

運行測試饺著,結(jié)果如下圖所示,針對 @ValueSource 里每個參數(shù)都會運行目標方法肠牲,一旦哪個參數(shù)運行測試失敗瓶籽,就意味著該測試方法不通過。

image

CSV 數(shù)據(jù)源測試:@CsvSource

通過 @CsvSource 可以注入指定 CSV 格式 (comma-separated-values) 的一組數(shù)據(jù)埂材,用每個逗號分隔的值來匹配一個測試方法對應(yīng)的參數(shù),下面是使用示例:

@ParameterizedTest
@CsvSource({"1,One", "2,Two", "3,Three"})
void testDataFromCsv(long id, String name) {
    System.out.printf("id: %d, name: %s", id, name);
}

運行結(jié)果如圖所示汤求,除了用逗號分隔參數(shù)外俏险,@CsvSource 還支持自定義符號,只要修改它的 delimiter 即可扬绪,默認為 竖独,

image

JUnit 還提供了讀取外部 CSV 格式文件數(shù)據(jù)的方式作為數(shù)據(jù)源的實現(xiàn)挤牛,我們只要用 @CsvFileSource 指定資源文件路徑即可莹痢,使用起來跟 @CsvSource 一樣簡單這里就不再重復(fù)演示了。

@CsvFileSource 指定的資源文件路徑時要以 / 開始,尋找當前測試資源目錄下文件竞膳。

除了上面提到的三種數(shù)據(jù)源方式外航瞭,JUnit 還提供了以下三種數(shù)據(jù)源:

  • @EnumSource:允許我們通過參數(shù)值,給指定 Enum 枚舉類型傳入坦辟,構(gòu)造出枚舉類型中特定的值刊侯。
  • @MethodSource:指定一個返回的 Stream / Array / 可迭代對象 的方法作為數(shù)據(jù)源。 需要注意的是該方法必須是靜態(tài)的锉走,并且不能接受任何參數(shù)滨彻。
  • @ArgumentSource:通過實現(xiàn) ArgumentsProvider 接口的參數(shù)類來作為數(shù)據(jù)源,重寫它的 provideArguments 方法可以返回自定義類型的 Stream<Arguments> 挪蹭,作為測試方法所需要的數(shù)據(jù)使用亭饵。

對上面三種數(shù)據(jù)源注解感興趣的同學可以參考示例工程的 ParameterizedUnitTest 類,這里就不一一再介紹了梁厉。

參考資料

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末辜羊,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子懂算,更是在濱河造成了極大的恐慌只冻,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件计技,死亡現(xiàn)場離奇詭異喜德,居然都是意外死亡,警方通過查閱死者的電腦和手機垮媒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進店門舍悯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人睡雇,你說我怎么就攤上這事萌衬。” “怎么了它抱?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵秕豫,是天一觀的道長。 經(jīng)常有香客問我观蓄,道長混移,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任侮穿,我火速辦了婚禮歌径,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘亲茅。我一直安慰自己回铛,他們只是感情好狗准,可當我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著茵肃,像睡著了一般腔长。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上免姿,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天饼酿,我揣著相機與錄音,去河邊找鬼胚膊。 笑死故俐,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的紊婉。 我是一名探鬼主播药版,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼喻犁!你這毒婦竟也來了槽片?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤肢础,失蹤者是張志新(化名)和其女友劉穎还栓,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體传轰,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡剩盒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了慨蛙。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片辽聊。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖期贫,靈堂內(nèi)的尸體忽然破棺而出跟匆,到底是詐尸還是另有隱情,我是刑警寧澤通砍,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布玛臂,位于F島的核電站,受9級特大地震影響封孙,放射性物質(zhì)發(fā)生泄漏垢揩。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一敛瓷、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧斑匪,春花似錦呐籽、人聲如沸锋勺。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽庶橱。三九已至,卻和暖如春贪惹,著一層夾襖步出監(jiān)牢的瞬間苏章,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工奏瞬, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留枫绅,地道東北人。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓硼端,卻偏偏與公主長得像并淋,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子珍昨,可洞房花燭夜當晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容