Java正則表達式的應用

在不少種狀況下,咱們都必須對字符串進行匹配,以便判斷字符串的格式是否符合要求,對字符串中的內容進行提取。好比,我要從一段話aabdfe中,判斷這段話是否有包含ab這個詞,那麼若是用if-else來判斷的話,那麼咱們必須遍歷整個字符串,當遇到一個a,記錄一下狀態,判斷下一個是不是所要的b。這個過程隨着要判斷的內容(在這裏是ab)和要被字符串的長度的增加,噁心程度遞增。可是又由於字符串的判斷實在是太常要用到啦,因此就有了正則表達式這麼個東西,正則表達式其實就是一個字符串識別的規則,經過這個規則,咱們就可讓程序根據這個規則去識別了。在Java裏面使用正則表達式須要涉及到兩個PatternMatcher。Pattern和Matcher之間的關係就比如Pattern是作模具的師傅,pattern將模具(正則表達)作好以後,指派一個小工(matcher)去匹配,matcher要作的就是原材料(即要被匹配的源字符串)和模具(即Pattern中的正則表達式)配對、比較。本文將探討跟Java正則表達式有關的一些應用,並嘗試着經過代碼說明Pattern和Matcher的用法。html

--對於正則表達式不熟悉的同窗,請自行到《正則表達式30分鐘入門教程》中學習。java

一、Matcher中的分組


    使用Matcher類,咱們必需要先了解一下正則表達式中分組(Group)的概念不清楚組的概念的同窗,請自行到正則表達式30分鐘入門教程中的分組一節學習。簡單的說,分組其實就是爲了可以指定同一個規則可使用多少次,有點像咱們買蘋果,假設咱們要買6顆蘋果,那麼咱們能夠用連續使用6次又大又紅的規則來挑6個蘋果。正則表達式中的分組是就是整個大的正則表達式和用()圈起來的內容。下面舉一個例子。面試

在這個正則表達式"\\w(\\d\\d)(\\w+)"中,正則表達式

分組0:是"\\w(\\d\\d)(\\w+)"

分組1:是(\\d\\d)

分組2:是(\\w+)

若是咱們稍稍變換一下,將原先的正則表達式改成"(\\w)(\\d\\d)(\\w+)"api

在這個正則表達式"\\w(\\d\\d)(\\w+)"中,

分組0:是"\\w(\\d\\d)(\\w+)"

分組1:是(\\d\\d)

分組2:是(\\w+)

 

咱們看看和正則表達式」\\w(\\d\\d)(\\w+)」匹配的一個字符串x99SuperJava, 
group(0)是匹配整個表達式的字符串的那部分A22happy 
group(1)是第1組(\d\d)匹配的部分:22 
group(2)是第二組(\w+)匹配的那部分happyoracle

讀者也但是用下面的代碼驗證一下app

 

public static void main(String[] args) {
        String Regex="\\w(\\d\\d)(\\w+)";
        String TestStr="A22happy";
        Pattern p=Pattern.compile(Regex);
        Matcher matcher=p.matcher(TestStr);
        if (matcher.find()) {
            int gc=matcher.groupCount();
            for (int i = 0; i <= gc; i++) {
                System.out.println("group "+i+" :"+matcher.group(i));
            }
        }
    }

 

記得要引用工具

import java.util.regex.Matcher;
import java.util.regex.Pattern;

二、Matcher經常使用方法

public Matcher reset()

這個方法將Matcher的狀態從新設置爲最初的狀態。學習

public Matcher reset(CharSequence input)

從新設置Matcher的狀態,而且將候選字符序列設置爲input後進行Matcher, 這個方法和從新建立一個Matcher同樣,只是這樣能夠重用之前的對象。測試

public int start()

這個方法返回了,Matcher所匹配的字符串在整個字符串的的開始下標:

下面咱們用一個小例子說明Matcher.start的用處

 

public static void testStart(){
         //建立一個 Matcher ,使用 Matcher.start()方法
         String candidateString = "My name is Bond. James Bond.";
         String matchHelper[] =
          {"          ^","                      ^"};
         Pattern p = Pattern.compile("Bond");
         Matcher matcher = p.matcher(candidateString);
         //找到第一個 'Bond'的開始下標
          matcher.find();
          int startIndex = matcher.start();
          System.out.println(candidateString);
          System.out.println(matchHelper[0] + startIndex);
         //找到第二個'Bond'的開始下標
          matcher.find();
          int nextIndex = matcher.start();
          System.out.println(candidateString);
          System.out.println(matchHelper[1] + nextIndex);
        
    }

 

結果截圖:

image

 

public int start(int group)

這個方法能夠指定你感興趣的sub group,而後返回sup group(子分組)匹配的開始位置。

public int end()

這個和start()對應,返回在之前的匹配操做期間,由給定組所捕獲子序列的最後字符以後的偏移量。 
其實start和end常常是一塊兒配合使用來返回匹配的子字符串。

public int end(int group)

和public int start(int group)對應,返回在sup group(子分組)匹配的子字符串最後一個匹配字符的位置。

public String group()

返回由之前匹配操做所匹配的字符串。 
這個方法提供了強大而方便的工具,他能夠等同使用start和end,而後對字符串做substring(start,end)操做。

看看下面一個小例子:

 

/**
     * 測試matcher.group方法
     */
    public static void testGroup() {
        // 建立一個 Pattern
        Pattern p = Pattern.compile("Bond");
        // 建立一個 Matcher ,以便使用 Matcher.group() 方法
        String candidateString = "My name is Bond. James Bond.";
        Matcher matcher = p.matcher(candidateString);
        // 提取 group
        matcher.find();
        System.out.println(String.format("group匹配的字符串 : %s",matcher.group()));
        System.out.println(String.format("匹配的開始位置 : %d", matcher.start()));
        System.out.println(String.format("匹配的結束位置 : %d", matcher.end()));

        System.out
                .println("---再次使用matcher.find()方法,看看matcher中group、start、end方法的效果");
        matcher.find();
        System.out.println(String.format("group匹配的字符串 : %s",matcher.group()));;
        System.out.println(String.format("匹配的開始位置 : %d", matcher.start()));
        System.out.println(String.format("匹配的結束位置 : %d", matcher.end()));
        System.out.println(String.format("candidateString字符串的長度 : %d", candidateString.length()));
    }

 

結果截圖:

image

三、最後來一個正則表達式的面試題來結束

1.判斷身份證:要麼是15位,要麼是18位,最後一位能夠爲字母,並寫程序提出其中的年月日。

 

public static void main(String[] args) {
        testID_Card();
    }

    public static void testID_Card() {
        // 測試是否爲合法的身份證號碼
        String[] strs = { "130681198712092019", "13068119871209201x",
                "13068119871209201", "123456789012345", "12345678901234x",
                "1234567890123" };
        // 準備正則表達式(身份證有15位和18位兩種,身份證的最後一位多是字母)
        String regex = "(\\d{14}\\w)|\\d{17}\\w";
        // 準備開始匹配,判斷全部的輸入是不是正確的
        Pattern regular = Pattern.compile(regex); // 建立匹配的規則Patter

        StringBuilder sb = new StringBuilder();
        // 遍歷全部要匹配的字符串
        for (int i = 0; i < strs.length; i++) {

            Matcher matcher = regular.matcher(strs[i]);// 建立一個Matcher
            sb.append("身份證:  ");
            sb.append(strs[i]);
            sb.append("   匹配:");
            sb.append(matcher.matches());
            System.out.println(sb.toString());
            sb.delete(0, sb.length());// 清空StringBuilder的方法
        }

        GetBirthDay(strs);

    }

    private static void GetBirthDay(String[] strs) {
        System.out.println("準備開始獲取出生日期");
        // 準備驗證規則
        Pattern BirthDayRegular = Pattern.compile("(\\d{6})(\\d{8})(.*)");
        // .*連在一塊兒就意味着任意數量的不包含換行的字符
        Pattern YearMonthDayRegular = Pattern
                .compile("(\\d{4})(\\d{2})(\\d{2})");
        for (int i = 0; i < strs.length; i++) {
            Matcher matcher = BirthDayRegular.matcher(strs[i]);

            if (matcher.matches()) {
                Matcher matcher2 = YearMonthDayRegular
                        .matcher(matcher.group(2));
                if (matcher2.matches()) {
                    System.out.println(strs[i]+"    中的出生年月分解爲: "+"年" + matcher2.group(1) + "   月:"
                            + matcher2.group(2) + "  日:" + matcher2.group(3));

                }
            }
        }

    }

 

結果截圖:

image

相關文章
相關標籤/搜索