Java正則表達式API邊界匹配
Boundary Matchers
Java regex API還支持邊界匹配。如果我們關心在輸入文本中匹配的確切位置,那麼這就是我們要尋找的。在前面的示例中,我們關心的隻是是否找到匹配項。
為瞭僅在文本開頭所需的正則表達式為true時匹配,我們使用插入符號^。
此測試將失敗,因為可以在開頭找到文本dog:
@Test public void givenText_whenMatchesAtBeginning_thenCorrect() { int matches = runTest("^dog", "dogs are friendly"); assertTrue(matches > 0); }
下面的測試將失敗:
@Test public void givenTextAndWrongInput_whenMatchFailsAtBeginning_ thenCorrect() { int matches = runTest("^dog", "are dogs are friendly?"); assertFalse(matches > 0); }
為瞭僅在文本末尾所需的正則表達式為true時匹配,我們使用美元字符$
。在以下情況下會找到匹配項:
@Test public void givenText_whenMatchesAtEnd_thenCorrect() { int matches = runTest("dog$", "Man's best friend is a dog"); assertTrue(matches > 0); }
並且沒有找到匹配:
@Test public void givenTextAndWrongInput_whenMatchFailsAtEnd_thenCorrect() { int matches = runTest("dog$", "is a dog man's best friend?"); assertFalse(matches > 0); }
如果僅在單詞邊界處找到所需文本時才需要匹配,則在正則表達式的開頭和結尾使用\\b
正則表達式:
空格是單詞邊界:
@Test public void givenText_whenMatchesAtWordBoundary_thenCorrect() { int matches = runTest("\\bdog\\b", "a dog is friendly"); assertTrue(matches > 0); }
行首的空字符串也是單詞邊界:
@Test public void givenText_whenMatchesAtWordBoundary_thenCorrect2() { int matches = runTest("\\bdog\\b", "dog is man's best friend"); assertTrue(matches > 0); }
這些測試之所以通過,是因為字符串的開頭以及文本之間的空格標記瞭單詞邊界,但是以下測試顯示瞭相反的結果:
@Test public void givenWrongText_whenMatchFailsAtWordBoundary_thenCorrect() { int matches = runTest("\\bdog\\b", "snoop dogg is a rapper"); assertFalse(matches > 0); }
一行中出現的兩個單詞字符不會標記單詞邊界,但我們可以通過更改正則表達式的結尾來查找非單詞邊界:
@Test public void givenText_whenMatchesAtWordAndNonBoundary_thenCorrect() { int matches = runTest("\\bdog\\B", "snoop dogg is a rapper"); assertTrue(matches > 0); }
Pattern Class Methods
之前,我們隻以基本方式創建瞭模式對象。然而,這個類有另一個compile方法的變體,它接受一組標志以及影響模式匹配方式的regex參數。
這些標志隻是抽象的整數值。讓我們重載test類中的runTest方法,以便它可以將標志作為第三個參數:
public static int runTest(String regex, String text, int flags) { pattern = Pattern.compile(regex, flags); matcher = pattern.matcher(text); int matches = 0; while (matcher.find()){ matches++; } return matches; }
在本節中,我們將瞭解不同的支持標志以及它們的使用方式。
Pattern.CANON_EQ
此標志啟用canonical equivalence
,當且僅當兩個字符的完整規范分解匹配時,才會認為這兩個字符匹配。
考慮帶重音的Unicode字符é
。它的復合代碼點是u00E9
。但是,Unicode的組成字符e
、u0065
和u0301
也有單獨的代碼點。在這種情況下,合成字符u00E9
與雙字符序列u0065 u0301
無法區分。
默認情況下,匹配不考慮規范等效:
@Test public void givenRegexWithoutCanonEq_whenMatchFailsOnEquivalentUnicode_thenCorrect() { int matches = runTest("\u00E9", "\u0065\u0301"); assertFalse(matches > 0); }
但如果添加標志,則測試將通過:
@Test public void givenRegexWithCanonEq_whenMatchesOnEquivalentUnicode_thenCorrect() { int matches = runTest("\u00E9", "\u0065\u0301", Pattern.CANON_EQ); assertTrue(matches > 0); }
Pattern.CASE_INSENSITIVE
無論大小寫,此標志都支持匹配。默認情況下,匹配會考慮大小寫:
@Test public void givenRegexWithDefaultMatcher_whenMatchFailsOnDifferentCases_thenCorrect() { int matches = runTest("dog", "This is a Dog"); assertFalse(matches > 0); }
因此,使用此標志,我們可以更改默認行為:
@Test public void givenRegexWithCaseInsensitiveMatcher _whenMatchesOnDifferentCases_thenCorrect() { int matches = runTest( "dog", "This is a Dog", Pattern.CASE_INSENSITIVE); assertTrue(matches > 0); }
我們還可以使用等效的嵌入標志表達式來實現相同的結果:
@Test public void givenRegexWithEmbeddedCaseInsensitiveMatcher _whenMatchesOnDifferentCases_thenCorrect() { int matches = runTest("(?i)dog", "This is a Dog"); assertTrue(matches > 0); }
Pattern.COMMENTS
Java API允許在正則表達式中包含使用#的註釋。這有助於記錄復雜的正則表達式,而其他程序員可能無法立即看到這些正則表達式。
comments標志使matcher忽略正則表達式中的任何空白或註釋,隻考慮模式。
在默認匹配模式下,以下測試將失敗:
@Test public void givenRegexWithComments_whenMatchFailsWithoutFlag_thenCorrect() { int matches = runTest( "dog$ #check for word dog at end of text", "This is a dog"); assertFalse(matches > 0); }
這是因為匹配器將在輸入文本中查找整個正則表達式,包括空格和#
字符。但當我們使用該標志時,它將忽略額外的空格,並且以#
開頭的每個文本都將被視為每行要忽略的註釋:
@Test public void givenRegexWithComments_whenMatchesWithFlag_thenCorrect() { int matches = runTest( "dog$ #check end of text","This is a dog", Pattern.COMMENTS); assertTrue(matches > 0); }
還有一個替代的嵌入式標志表達式:
@Test public void givenRegexWithComments_whenMatchesWithEmbeddedFlag_thenCorrect() { int matches = runTest( "(?x)dog$ #check end of text", "This is a dog"); assertTrue(matches > 0); }
到此這篇關於Java正則表達式API邊界匹配的文章就介紹到這瞭,更多相關Java正則表達式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 如何在Java中使用正則表達式API
- Java正則表達式API字符類
- Java中Pattern.compile函數的使用詳解
- Java之Pattern.compile函數用法詳解
- Java正則表達式之Pattern類實例詳解