Java正则表达式的应用

简介:

在很多种情况下,我们都必须对字符串进行匹配,以便判断字符串的格式是否符合要求,对字符串中的内容进行提取。比如,我要从一段话aabdfe中,判断这段话是否有包含ab这个词,那么如果用if-else来判断的话,那么我们必须遍历整个字符串,当遇到一个a,记录一下状态,判断下一个是否是所要的b。这个过程随着要判断的内容(在这里是ab)和要被字符串的长度的增长,恶心程度递增。但是又因为字符串的判断实在是太常要用到啦,所以就有了正则表达式这么个东西,正则表达式其实就是一个字符串识别的规则,通过这个规则,我们就可以让程序根据这个规则去识别了。在Java里面使用正则表达式需要涉及到两个PatternMatcher。Pattern和Matcher之间的关系就好比Pattern是做模具的师傅,pattern将模具(正则表达)做好之后,指派一个小工(matcher)去匹配,matcher要做的就是原材料(即要被匹配的源字符串)和模具(即Pattern中的正则表达式)配对、比较。本文将探讨跟Java正则表达式有关的一些应用,并尝试着通过代码说明Pattern和Matcher的用法。

--对于正则表达式不熟悉的同学,请自行到《正则表达式30分钟入门教程》中学习。

1、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+)"

我们的分组就变成了

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

分组1:是"(\\w)"

分组2:是"(\\d\\d)"

分组3:是"(\\w+)"

 

我们看看和正则表达式”\\w(\\d\\d)(\\w+)”匹配的一个字符串x99SuperJava, 
group(0)是匹配整个表达式的字符串的那部分A22happy 
group(1)是第1组(\d\d)匹配的部分:22 
group(2)是第二组(\w+)匹配的那部分happy

读者也可是用下面的代码验证一下

复制代码
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;

2、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

3、最后来一个正则表达式的面试题来结束

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

 

 

参考链接

Java中正则Matcher类的matches()、lookAt()和find()的区别

学习正则表达式:Matcher类

本文转自陈哈哈博客园博客,原文链接http://www.cnblogs.com/kissazi2/p/3287206.html如需转载请自行联系原作者

kissazi2
相关文章
|
1天前
|
Java 数据处理 开发者
Java中的Lambda表达式:简化你的代码之路
【8月更文挑战第66天】Lambda表达式在Java 8中首次引入,它为Java开发者提供了一种更简洁、更灵活的编程方式。本文将通过简单易懂的语言和实际代码示例,引导你理解Lambda表达式的基本概念、语法结构以及如何在Java项目中应用它来简化代码。无论你是Java新手还是有经验的开发者,这篇文章都将帮助你更好地掌握这一强大的工具。
28 11
|
1天前
|
Java API
Java中的Lambda表达式:一种简洁而强大的编程工具
本文将深入探讨Java中的Lambda表达式,它是一种简洁而强大的编程工具。通过本文,您将了解Lambda表达式的基本概念、语法结构以及使用方法。我们还将介绍Lambda表达式的应用场景和最佳实践,并通过实例演示其在简化代码、提高可读性和编写并行代码方面的优势。最后,我们将讨论使用Lambda表达式时需要注意的一些事项和限制。无论您是初学者还是有经验的Java开发人员,都可以从本文中获得有价值的知识和技能。现在,让我们一起探索Java中Lambda表达式的奇妙世界吧!
|
7天前
|
Java
探索Java中的Lambda表达式
【9月更文挑战第28天】Lambda表达式,这一Java 8引入的轻量级语法结构,为函数式编程提供了强有力的支持。它简化了代码,提高了开发效率,并使得并行处理数据变得异常容易。本文将通过直观的例子和详细的解释,带领读者深入理解Lambda表达式的奥秘,从而在编程实践中更加灵活地运用它们。
|
8天前
|
Java API
探索Java中的Lambda表达式
【9月更文挑战第27天】Lambda表达式,作为Java 8的一大亮点,为我们的编程世界带来了一股清新之风。它以其简洁明了的语法和强大的功能,让我们在处理集合、线程等复杂操作时如鱼得水。本文将带你一探Lambda表达式的奥秘,让你领略到它的魅力所在。
18 6
|
9天前
|
Java API 开发者
探索Java中的Lambda表达式:简洁与强大的代码实践
本文深入探讨Java中Lambda表达式的定义、用法及优势,通过实例展示其如何简化代码、提升可读性,并强调在使用中需注意的兼容性和效率问题。Lambda作为Java 8的亮点功能,不仅优化了集合操作,还促进了函数式编程范式的应用,为开发者提供了更灵活的编码方式。
|
5天前
|
Java 开发者
探索Java中的Lambda表达式:简化你的代码之旅##
【8月更文挑战第62天】 Java 8的发布为开发者带来了诸多新特性,其中最引人注目的无疑是Lambda表达式。这一特性不仅让代码变得更加简洁,还极大地提升了开发的效率。本文将通过实际示例,展示如何利用Lambda表达式来优化我们的代码结构,同时探讨其背后的工作原理和性能考量。 ##
|
6天前
|
Java API 开发者
探索Java中的Lambda表达式
【9月更文挑战第29天】Lambda表达式,在Java编程中,它不仅仅是一种语法糖,更是简化代码、提升效率的利器。本文将通过浅显易懂的语言和实际代码示例,带你走进Lambda的世界,了解其背后的原理和在日常开发中的应用,让你的代码更简洁、更高效。
|
8天前
|
Java API 开发者
探索Java中的Lambda表达式:简化代码,提升效率
【9月更文挑战第27天】在Java 8中引入的Lambda表达式为编程带来了革命性的变化。通过简洁的语法和强大的功能,它不仅简化了代码编写过程,还显著提升了程序的执行效率。本文将深入探讨Lambda表达式的本质、用法和优势,并结合实例演示其在实际开发中的应用。无论你是Java新手还是资深开发者,都能从中获得启发,优化你的代码设计。
|
7天前
|
Java 测试技术
Java接口的生产环境应用注意点
在Java生产环境中,合理使用接口对提升代码质量至关重要。设计接口时应遵循单一职责原则,采用清晰命名,并控制方法数量。默认方法应谨慎使用,避免与实现类产生冲突。通过版本化管理接口更新,确保向后兼容。实现接口时需明确行为,保持实现与接口分离,利用多态增强灵活性。关注性能影响,适当文档注释及充分测试确保接口稳定可靠。综合运用这些策略,可以显著提高系统的可扩展性和维护性。
|
10天前
|
JavaScript 前端开发 Java
Java 8 新特性详解及应用示例
Java 8 新特性详解及应用示例