自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JUnit 5系列之基礎(chǔ)入門介紹

開發(fā) 后端
本系列文章都基于 Junit 5發(fā)布的先行版 Milestone 2。它可能會(huì)有變化。如果有新的里程碑(milestone)版本發(fā)布,或者試用版正式發(fā)行時(shí),我會(huì)再來(lái)更新這篇文章。這里要介紹的多數(shù)知識(shí)你都可以在 JUnit 5 用戶指南 中找到(這個(gè)鏈接指向的是先行版 Milestone 2,想看的最新版本文檔的話請(qǐng)戳這里),并且指南還有更多的內(nèi)容等待你發(fā)掘。

上周我們剛剛搭建好了 JUnit 5 的環(huán)境,現(xiàn)在我們可以寫測(cè)試了。這節(jié)就讓我們來(lái)寫它幾個(gè)吧!

概述

本文章是這個(gè) JUnit 5 系列的一部分:

(如果不喜歡看文章,你可以戳這里看我的演講,或者看一下最近的 vJUG 講座,或者我在 DevoxxPL 上的 PPT。

本系列文章都基于 Junit 5發(fā)布的先行版 Milestone 2。它可能會(huì)有變化。如果有新的里程碑(milestone)版本發(fā)布,或者試用版正式發(fā)行時(shí),我會(huì)再來(lái)更新這篇文章。

這里要介紹的多數(shù)知識(shí)你都可以在 JUnit 5 用戶指南 中找到(這個(gè)鏈接指向的是先行版 Milestone 2,想看的***版本文檔的話請(qǐng)戳這里),并且指南還有更多的內(nèi)容等待你發(fā)掘。下面的所有代碼都可以在 我的 Github 上找到。

目錄

  • 設(shè)計(jì)哲學(xué)
  • JUnit 5預(yù)備:

。包可見性

。測(cè)試的生命周期

# @Test

# Before 和 After

# 禁用測(cè)試

# 測(cè)試類的生命周期

。斷言

# 常規(guī)斷言

# 擴(kuò)展斷言

。假言/判定(Assumptions)

。測(cè)試嵌套

。測(cè)試命名

  • 回顧
  • 分享&關(guān)注

設(shè)計(jì)哲學(xué)

新的架構(gòu)設(shè)計(jì)(這個(gè)我們?nèi)蘸罅?,其關(guān)注點(diǎn)在高擴(kuò)展性。如果后面出現(xiàn)了什么神之測(cè)試技術(shù)(至少對(duì)我們廣大 Java?來(lái)說(shuō)很神的),它們也可能在 JUnit 5 的架構(gòu)下被實(shí)現(xiàn)。

不過(guò)當(dāng)前來(lái)說(shuō),涉及的基礎(chǔ)知識(shí)與 JUnit 4 是非常相似的。JUnit 5 的改動(dòng)并不激進(jìn),相反它的優(yōu)化歷程是小心翼翼,小步迭代的。因此,開發(fā)者應(yīng)該會(huì)對(duì)新的 API 感到非常熟悉。至少我是這樣的,我相信你也不會(huì)感覺陌生: 

  1. class Lifecycle { 
  2.   
  3.     @BeforeAll 
  4.     static void initializeExternalResources() { 
  5.         System.out.println("Initializing external resources..."); 
  6.     } 
  7.   
  8.     @BeforeEach 
  9.     void initializeMockObjects() { 
  10.         System.out.println("Initializing mock objects..."); 
  11.     } 
  12.   
  13.     @Test 
  14.     void someTest() { 
  15.         System.out.println("Running some test..."); 
  16.         assertTrue(true); 
  17.     } 
  18.   
  19.     @Test 
  20.     void otherTest() { 
  21.         assumeTrue(true); 
  22.   
  23.         System.out.println("Running another test..."); 
  24.         assertNotEquals(1, 42, "Why wouldn't these be the same?"); 
  25.     } 
  26.   
  27.     @Test 
  28.     @Disabled 
  29.     void disabledTest() { 
  30.         System.exit(1); 
  31.     } 
  32.   
  33.     @AfterEach 
  34.     void tearDown() { 
  35.         System.out.println("Tearing down..."); 
  36.     } 
  37.   
  38.     @AfterAll 
  39.     static void freeExternalResources() { 
  40.         System.out.println("Freeing external resources..."); 
  41.     } 
  42.   
  43.  

是吧?這里并沒(méi)有很大的改動(dòng)。

JUnit 5 預(yù)備

包可見性

JUnit 5 最明顯的變化應(yīng)該是,不再需要手動(dòng)將測(cè)試類與測(cè)試方法為 public 了。包可見的訪問(wèn)級(jí)別就足夠了。當(dāng)然,私有(private)訪問(wèn)還是不行的。我認(rèn)為這個(gè)變化是合理的,也符合我們對(duì)可見性的一般直覺。

這很好!至少可以少打幾個(gè)字母了。不過(guò),我相信你也不是每次都手打這幾個(gè)字母的,是吧?盡管如此還是很好,少一些關(guān)鍵字,你在看測(cè)試的時(shí)候也少些切換。

測(cè)試的生命周期

@Test

JUnit 中最基本的注解非 @Test 莫屬了。它會(huì)標(biāo)記方法為測(cè)試方法,以便構(gòu)建工具和 IDE 能夠識(shí)別并執(zhí)行它們。

它的 API 和作用并沒(méi)有變化,不過(guò)它不再接受任何參數(shù)了。若要測(cè)試是否拋出異常,你可以通過(guò)新的斷言 API 來(lái)做到;不過(guò)就我所知,目前還沒(méi)有超時(shí)選項(xiàng)timeout的替代品。

與 JUnit 4一樣,JUnit 5 會(huì)為每個(gè)測(cè)試方法創(chuàng)建一個(gè)新的實(shí)例。

Before 和 After

你可能需要執(zhí)行一些代碼來(lái)在測(cè)試執(zhí)行前后完成一些初始化或銷毀的操作。在 JUnit 5 中,有4個(gè)注解你可能會(huì)用于如此工作:

@BeforeAll

只執(zhí)行一次,執(zhí)行時(shí)機(jī)是在所有測(cè)試和 @BeforeEach 注解方法之前。

@BeforeEach

在每個(gè)測(cè)試執(zhí)行之前執(zhí)行。

@AfterEach

在每個(gè)測(cè)試執(zhí)行之后執(zhí)行。

@AfterAll

只執(zhí)行一次,執(zhí)行時(shí)機(jī)是在所有測(cè)試和 @AfterEach 注解方法之后。

因?yàn)榭蚣軙?huì)為每個(gè)測(cè)試創(chuàng)建一個(gè)單獨(dú)的實(shí)例,在 @BeforeAll/@AfterAll 方法執(zhí)行時(shí)尚無(wú)任何測(cè)試實(shí)例誕生。因此,這兩個(gè)方法必須定義為靜態(tài)方法。

注解了同樣一個(gè)注解的不同方法,其執(zhí)行次序是不可預(yù)知的,包括對(duì)繼承來(lái)的方法也適用。這是開發(fā)團(tuán)隊(duì)經(jīng)過(guò)審慎思考后的決定,即把單元測(cè)試與集成測(cè)試的關(guān)注點(diǎn)分開。集成測(cè)試可能需要方法間更緊密的協(xié)作,但一個(gè)單元測(cè)試不應(yīng)該對(duì)其他的單元測(cè)試有所依賴。而對(duì)于集成測(cè)試——也叫場(chǎng)景測(cè)試——的支持,也已在團(tuán)隊(duì)的計(jì)劃中。

除了名字有所不同,這幾個(gè)注解與 JUnit 4 中的注解工作方式完全一樣。無(wú)獨(dú)有偶,跟主流意見一致,我也覺得這個(gè)新的命名不能說(shuō)服我其必要性。這個(gè) issue 下有更多的討論。

禁用測(cè)試

今兒星期五,抬頭一看已經(jīng)4點(diǎn)半,無(wú)心工作的你想回家了?完全理解,在測(cè)試上怒拍一個(gè) @Disabled 注解即可。有良心的話寫個(gè)忽略測(cè)試的理由是極好的,不過(guò)也可以不帶此參數(shù)。 

  1. @Test  
  2. @Disabled("你丫就是存心跑不過(guò)的是不?!"
  3. void failingTest() { 
  4.     assertTrue(false); 
  5.  

測(cè)試類的生命周期

JUnit 團(tuán)隊(duì)發(fā)布的***版原型中,包含了一個(gè)對(duì) 測(cè)試類的生命周期 的描述,有意思的是,這個(gè)特性在 alpha 版本的發(fā)布中未被加入。這個(gè)生命周期模型建議,在被測(cè)類的多個(gè)測(cè)試方法中使用一個(gè)同樣的實(shí)例,因?yàn)檫@樣我們就可以通過(guò)改變對(duì)象的狀態(tài),進(jìn)而實(shí)現(xiàn)在多個(gè)測(cè)試方法中的交互。(我也再說(shuō)一遍,這更像是 場(chǎng)景測(cè)試 要管的事。)

正如我在***版公測(cè)時(shí)所說(shuō),這樣的特性99%的場(chǎng)景下是有害的,只有另外1%的場(chǎng)合下才有真正的用處。我只能說(shuō),還好這個(gè)特性被摒棄了。想想你的單元測(cè)試,如果它們必須靠在方法間維護(hù)狀態(tài)來(lái)工作,這畫面簡(jiǎn)直太美我不敢看?。

斷言

如果說(shuō) @Test、@Before...、@After... 等注解是一個(gè)測(cè)試套件的骨架,那么斷言就是它的心臟。準(zhǔn)備好測(cè)試實(shí)例、執(zhí)行了被測(cè)類的方法以后,斷言能確保你得到了想要的結(jié)果。否則,就說(shuō)明當(dāng)前測(cè)試失敗了。

常規(guī)斷言

一般的斷言,無(wú)非是檢查一個(gè)實(shí)例的屬性(比如,判空與判非空等),或者對(duì)兩個(gè)實(shí)例進(jìn)行比較(比如,檢查兩個(gè)實(shí)例對(duì)象是否相等)等。無(wú)論哪種檢查,斷言方法都可以接受一個(gè)字符串作為***一個(gè)可選參數(shù),它會(huì)在斷言失敗時(shí)提供必要的描述信息。如果提供出錯(cuò)信息的過(guò)程比較復(fù)雜,它也可以被包裝在一個(gè) lambda 表達(dá)式中,這樣,只有到真正失敗的時(shí)候,消息才會(huì)真正被構(gòu)造出來(lái)。 

  1. @Test 
  2. void assertWithBoolean() { 
  3.     assertTrue(true); 
  4.     assertTrue(this::truism); 
  5.   
  6.     assertFalse(false, () -> "Really " + "expensive " + "message" + "."); 
  7.   
  8. boolean truism() { 
  9.     return true
  10.   
  11. @Test 
  12. void assertWithComparison() { 
  13.     List<String> expected = asList("element"); 
  14.     List<String> actual = new LinkedList<>(expected); 
  15.   
  16.     assertEquals(expected, actual); 
  17.     assertEquals(expected, actual, "Should be equal."); 
  18.     assertEquals(expected, actual, () -> "Should " + "be " + "equal."); 
  19.      
  20.     assertNotSame(expected, actual, "Obviously not the same instance."); 
  21.  

如你所見,JUnit 5 的 API 并無(wú)太多變化。斷言方法的命名是一樣的,方法同樣接受兩個(gè)參數(shù),分別是一個(gè)期望值與一個(gè)實(shí)際值。

期望值與實(shí)際值的傳入順序非常重要,無(wú)論是對(duì)于理解測(cè)試的內(nèi)容,還是理解失敗時(shí)的錯(cuò)誤信息,但有時(shí)還是很容易弄錯(cuò),這點(diǎn)很坑。不過(guò)仔細(xì)想想,也沒(méi)什么更好的辦法,除非你自己創(chuàng)建一個(gè)新的斷言框架。既然市面上已有對(duì)應(yīng)的產(chǎn)品如 Hamcrest (ugh!) 和AssertJ (yeah!譯者表示:不太清楚這歡呼的梗在哪里)等,再浪費(fèi)有限的時(shí)間去造輪子明顯不值得。畢竟最重要的是保證你的斷言庫(kù)專注于一件事,借鑒已有實(shí)現(xiàn)可以節(jié)省成本。

哦對(duì)了,失敗信息現(xiàn)在是作為***傳入的參數(shù)了。我很喜歡這個(gè)細(xì)節(jié),因?yàn)?,它讓你專注于真正重要之?mdash;—那兩個(gè)需被斷言的值。由于擁抱了 Java 8 的緣故,真值斷言方法現(xiàn)在也接受 supplier 參數(shù)了,又是一個(gè)暖心的小細(xì)節(jié)。

擴(kuò)展斷言

除了那種一般的檢查特定實(shí)例或?qū)傩缘臄嘌酝?,還有一些其他類型的斷言。

這里要講的***個(gè)甚至都不是個(gè)真正的斷言,它做的事就是強(qiáng)行讓測(cè)試失敗,并提供一個(gè)失敗信息。 

  1. @Test 
  2. void failTheTest() { 
  3.     fail("epicly"); 
  4.  

還有 assertAll 方法,它接受可變數(shù)量的斷言作為參數(shù),并保證它們?nèi)康玫綀?zhí)行,然后再把錯(cuò)誤信息(如果有)一并匯報(bào)出來(lái)。 

  1. @Test 
  2. void assertAllProperties() { 
  3.     Address address = new Address("New City""Some Street""No"); 
  4.   
  5.     assertAll("address"
  6.             () -> assertEquals("Neustadt", address.city), 
  7.             () -> assertEquals("Irgendeinestraße", address.street), 
  8.             () -> assertEquals("Nr", address.number) 
  9.     ); 
  10.   
  1. org.opentest4j.MultipleFailuresError: address (3 failures) 
  2.     expected: <Neustadt> but was: <New City> 
  3.     expected: <Irgendeinestraße> but was: <Some Street> 
  4.     expected: <Nr> but was: <No 

這個(gè)特性在檢查對(duì)象的多個(gè)屬性值時(shí)非常有用。按照一般的做法,測(cè)試在***個(gè)斷言失敗時(shí)就會(huì)掛掉了,此時(shí)只有***個(gè)出錯(cuò)的地方得到提示,而你無(wú)法得知其他值的斷言是否成功,只好再跑一遍測(cè)試。

***,我們終于有了 assertThrows 和 expectThrows 方法。兩者均會(huì)在被測(cè)方法未拋出預(yù)期異常時(shí)失敗。而后者還會(huì)返回拋出的異常實(shí)例,以用于后續(xù)的驗(yàn)證,比如,斷言異常信息包含正確的信息等。 

  1. @Test 
  2. void assertExceptions() { 
  3.     assertThrows(Exception.class, this::throwing); 
  4.   
  5.     Exception exception = expectThrows(Exception.class, this::throwing); 
  6.     assertEquals("Because I can!", exception.getMessage()); 

 假言/判定(Assumptions)


假言/判定允許你僅在特定條件滿足時(shí)才運(yùn)行測(cè)試。這個(gè)特性能夠減少測(cè)試組件的運(yùn)行時(shí)間和代碼重復(fù),特別是在假言都不滿足的情況下。 

  1. @Test 
  2. void exitIfFalseIsTrue() { 
  3.     assumeTrue(false); 
  4.     System.exit(1); 
  5.   
  6. @Test 
  7. void exitIfTrueIsFalse() { 
  8.     assumeFalse(this::truism); 
  9.     System.exit(1); 
  10.   
  11. private boolean truism() { 
  12.     return true
  13.   
  14. @Test 
  15. void exitIfNullEqualsString() { 
  16.     assumingThat( 
  17.             "null".equals(null), 
  18.             () -> System.exit(1) 
  19.     ); 
  20.  

假言/判定適用于兩種情形,要么是你希望在某些條件不滿足時(shí)中止測(cè)試,要么是你希望僅當(dāng)某個(gè)條件滿足時(shí)才執(zhí)行(部分)測(cè)試。主要的區(qū)別是,被中止的測(cè)試是以被禁用(disabled)的形式被報(bào)告,此時(shí)沒(méi)有測(cè)試任何內(nèi)容,因?yàn)闂l件得不到滿足。

測(cè)試嵌套

在 JUnit 5 中,嵌套測(cè)試幾乎不費(fèi)吹灰之力。你只需要在嵌套的類上添加 @Nested 注解,類中的所有方法即會(huì)被引擎執(zhí)行: 

  1. package org.codefx.demo.junit5; 
  2.   
  3. import org.junit.jupiter.api.BeforeEach; 
  4. import org.junit.jupiter.api.Nested; 
  5. import org.junit.jupiter.api.Test; 
  6.   
  7. import static org.junit.jupiter.api.Assertions.assertEquals; 
  8. import static org.junit.jupiter.api.Assertions.assertTrue; 
  9.   
  10. class Nest { 
  11.      
  12.     int count = Integer.MIN_VALUE; 
  13.      
  14.     @BeforeEach 
  15.     void setCountToZero() { 
  16.         count = 0; 
  17.     } 
  18.      
  19.     @Test 
  20.     void countIsZero() { 
  21.         assertEquals(0, count); 
  22.     } 
  23.      
  24.     @Nested 
  25.     class CountGreaterZero { 
  26.   
  27.         @BeforeEach 
  28.         void increaseCount() { 
  29.             count++; 
  30.         } 
  31.   
  32.         @Test 
  33.         void countIsGreaterZero() { 
  34.             assertTrue(count > 0); 
  35.         } 
  36.   
  37.         @Nested 
  38.         class CountMuchGreaterZero { 
  39.   
  40.             @BeforeEach 
  41.             void increaseCount() { 
  42.                 count += Integer.MAX_VALUE / 2; 
  43.             } 
  44.   
  45.             @Test 
  46.             void countIsLarge() { 
  47.                 assertTrue(count > Integer.MAX_VALUE / 2); 
  48.             } 
  49.   
  50.         } 
  51.   
  52.     } 
  53.      

 如你所見,嵌套類中的 @BeforeEach(及 @AfterEach )注解也工作良好。不過(guò),構(gòu)造順序似乎還未被寫入文檔,它們的初始化次序是從外向內(nèi)的。這也讓你能疊加式地為內(nèi)部類準(zhǔn)備測(cè)試數(shù)據(jù)。

如果嵌套的內(nèi)部測(cè)試想要存取外部測(cè)試類的字段,那么嵌套類本身不應(yīng)該是靜態(tài)的。但這樣一來(lái)也就禁止了靜態(tài)方法的使用,因而這種場(chǎng)景下@BeforeAll 和 @AfterAll 方法也就無(wú)法使用了(還是說(shuō)終有他法實(shí)現(xiàn)?)

你可能有疑惑,嵌套的內(nèi)部測(cè)試類有什么用。個(gè)人而言,我用內(nèi)部類來(lái)漸進(jìn)測(cè)試接口,其他人則多用于保持測(cè)試類短小專注。后者同時(shí)也有一個(gè)經(jīng)典的例子來(lái)說(shuō)明,例子由 JUnit 團(tuán)隊(duì)提供,它測(cè)試了一個(gè)棧: 

  1. class TestingAStack { 
  2.   
  3.     Stack<Object> stack; 
  4.     boolean isRun = false
  5.   
  6.     @Test 
  7.     void isInstantiatedWithNew() { 
  8.         new Stack<Object>(); 
  9.     } 
  10.   
  11.     @Nested 
  12.     class WhenNew { 
  13.   
  14.         @BeforeEach 
  15.         void init() { 
  16.             stack = new Stack<Object>(); 
  17.         } 
  18.   
  19.         // some tests on 'stack', which is empty 
  20.   
  21.         @Nested 
  22.         class AfterPushing { 
  23.   
  24.             String anElement = "an element"
  25.   
  26.             @BeforeEach 
  27.             void init() { 
  28.                 stack.push(anElement); 
  29.             } 
  30.   
  31.             // some tests on 'stack', which has one element... 
  32.   
  33.         } 
  34.     } 
  35.  

在上面的例子中,棧的狀態(tài)改變會(huì)反映到內(nèi)層的測(cè)試類中,其中內(nèi)部類又基于自身的場(chǎng)景執(zhí)行了一些測(cè)試。

測(cè)試命名

JUnit 5 提供了一個(gè)注解 @DisplayName,它用以為開發(fā)者提供更可讀的測(cè)試類和測(cè)試方法信息。

上面的 stack 測(cè)試?yán)蛹由显撟⒔庖院缶妥兂蛇@樣: 

  1. @DisplayName("A stack"
  2. class TestingAStack { 
  3.   
  4.     @Test 
  5.     @DisplayName("is instantiated with new Stack()"
  6.     void isInstantiatedWithNew() { /*...*/ } 
  7.   
  8.     @Nested 
  9.     @DisplayName("when new"
  10.     class WhenNew { 
  11.   
  12.         @Test 
  13.         @DisplayName("is empty"
  14.         void isEmpty() { /*...*/ } 
  15.   
  16.         @Test 
  17.         @DisplayName("throws EmptyStackException when popped"
  18.         void throwsExceptionWhenPopped() { /*...*/ } 
  19.   
  20.         @Test 
  21.         @DisplayName("throws EmptyStackException when peeked"
  22.         void throwsExceptionWhenPeeked() { /*...*/ } 
  23.   
  24.         @Nested 
  25.         @DisplayName("after pushing an element"
  26.         class AfterPushing { 
  27.   
  28.             @Test 
  29.             @DisplayName("it is no longer empty"
  30.             void isEmpty() { /*...*/ } 
  31.   
  32.             @Test 
  33.             @DisplayName("returns the element when popped and is empty"
  34.             void returnElementWhenPopped() { /*...*/ } 
  35.   
  36.             @Test 
  37.             @DisplayName( 
  38.                     "returns the element when peeked but remains not empty"
  39.             void returnElementWhenPeeked(){ /*...*/ } 
  40.         } 
  41.     } 
  42.  

這是一份TDDer 看了會(huì)感動(dòng),BDDer 看了會(huì)流淚的測(cè)試結(jié)果輸出。

 

回顧

差不多就這些了,恭喜你終于讀完了。我們匆匆過(guò)完了 JUnit 5 的基本特性,現(xiàn)在,你應(yīng)該了解了所有寫測(cè)試的必備知識(shí)了:包括如何為方法添加生命周期注解(@[Before|After][All|Each]、如何注解測(cè)試方法本身(@Test)、如何嵌套測(cè)試(@Nested)、如何給測(cè)試一個(gè)好信息(@DisplayName),你也應(yīng)該能了解斷言和假言判定是如何工作的了(基本上與前版無(wú)異)。

不過(guò)這可還沒(méi)完!我們還沒(méi)聊到 測(cè)試方法的條件執(zhí)行,沒(méi)聊到非??岬?參數(shù)注入 ,以及 JUnit 5 的擴(kuò)展機(jī)制 和 架構(gòu)體系 呢。放心,這真的是***了,這些話題我們會(huì)一個(gè)月后再聊,現(xiàn)在你可以先休息一下啦。 

敬請(qǐng)期待下集!

責(zé)任編輯:龐桂玉 來(lái)源: segmentfault
相關(guān)推薦

2016-09-21 14:17:11

JUnit架構(gòu)API

2016-09-23 10:20:22

JUnit擴(kuò)展模型Extension

2011-04-18 11:13:45

2011-08-10 17:04:43

JavaScript

2021-09-16 10:05:09

鴻蒙HarmonyOS應(yīng)用

2022-02-20 07:28:13

Spring注解用法

2022-02-19 07:41:36

Bean注解項(xiàng)目

2010-05-27 17:41:09

2012-03-30 15:40:34

2024-01-03 07:57:11

高級(jí)參數(shù)PowerShellVerbose 參數(shù)

2023-09-21 07:06:17

PSDriveProvider

2021-03-02 12:36:49

MQKafkaRocketMQ

2009-07-07 14:33:44

JSP入門

2011-11-29 16:38:58

Knockout

2022-03-28 09:31:58

for循環(huán)語(yǔ)句

2014-10-11 09:16:59

互聯(lián)網(wǎng)協(xié)議

2021-05-18 07:15:37

Python

2018-07-11 13:33:43

大數(shù)據(jù)人工智能Hadoop

2019-10-12 15:06:02

MySQL數(shù)據(jù)庫(kù)命令

2023-05-29 16:25:59

Rust函數(shù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)