在很多种情况下,我们都必须对字符串进行匹配,以便判断字符串的格式是否符合要求,对字符串中的内容进行提取。比如,我要从一段话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); }

结果截图:

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()));
}

结果截图:

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)); }
}
} }

结果截图:

参考链接

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

学习正则表达式:Matcher类

Java正则表达式的应用的更多相关文章

  1. java正则表达式

    java正则表达式 1.Java正则表达式的语法与示例:  http://baike.xsoftlab.net/view/207.html 2.Java 正则表达式:  http://www.runo ...

  2. Java正则表达式入门——转自RUNOOB.COM

    Java 正则表达式 正则表达式定义了字符串的模式. 正则表达式可以用来搜索.编辑或处理文本. 正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别. Java正则表达式和Perl的是最为相似 ...

  3. Java 正则表达式详解

    Java 提供了功能强大的正则表达式API,在java.util.regex 包下.本教程介绍如何使用正则表达式API. 正则表达式 一个正则表达式是一个用于文本搜索的文本模式.换句话说,在文本中搜索 ...

  4. 【转】详解Java正则表达式语法

    (转自: http://www.jb51.net/article/76354.htm) 这篇文章主要介绍了Java正则表达式语法,包括常用正则表达式.匹配验证-验证Email是否正确以及字符串中查询字 ...

  5. java正则表达式【大全】

    [正则表达式]文本框输入内容控制整数或者小数:^[0-9]+\.{0,1}[0-9]{0,2}$只能输入数字:"^[0-9]*$".只能输入n位的数字:"^\d{n}$& ...

  6. JAVA正则表达式:Pattern类与Matcher类详解(转)

    java.util.regex是一个用正则表达式所订制的模式来对字符串进行匹配工作的类库包.它包括两个类:Pattern和Matcher Pattern 一个Pattern是一个正则表达式经编译后的表 ...

  7. JAVA正则表达式:Pattern类与Matcher类详解

    java.util.regex是一个用正则表达式所订制的模式来对字符串进行匹配工作的类库包.它包括两个类:Pattern和Matcher Pattern 一个Pattern是一个正则表达式经编译后的表 ...

  8. Java 正则表达式[转载]

    PS:转载自CSDN博客看上去很美 众所周知,在程序开发中,难免会遇到需要匹配.查找.替换.判断字符串的情况发生,而这些情况有时又比较复杂,如果用纯编码方式解决,往往会浪费程序员的时间及精力.因此,学 ...

  9. Java正则表达式实用教程

    java.util.regex是一个用正则表达式所订制的模式来对字符串进行匹配工作的类库包.java.util.regex包主要包括以下三个类:Pattern.Matcher和PatternSynta ...

随机推荐

  1. 《C和指针(Pointer on c)》 学习笔记

    转载:http://dsqiu.iteye.com/blog/1687944 首先本文是对参考中三个连接的博客进行的整理,非常感谢三位博主的努力,每次都感叹网友的力量实在太强大了…… 第一章 快速上手 ...

  2. Style file: generic.xaml

    All silverlight control style should be designed in generic.xaml which is in theme folder. But when ...

  3. Python-类的继承

    类的继承 面向对象的编程带来的主要好处之一是代码的重用,实现这种重用的方法之一是通过继承机制.继承完全可以理解成类之间的类型和子类型关系. 需要注意的地方:继承语法 class 派生类名(基类名):/ ...

  4. Android支付之支付宝封装类

    今天介绍下在android中如何集成支付宝支付到自己的APP中去.让APP能够拥有方便,快捷的支付功能. 我们在做Android支付的时候肯定会用到支付宝支付,根据官方给出的demo做起来非常费劲,所 ...

  5. (转)MVC,MVP 和 MVVM 的图示

    作者: 阮一峰 日期: 2015年2月 1日 复杂的软件必须有清晰合理的架构,否则无法开发和维护. MVC(Model-View-Controller)是最常见的软件架构之一,业界有着广泛应用.它本身 ...

  6. 初识ASP.NET CORE:二、优劣

    Which one is right for me? ASP.NET is a mature web platform that provides all the services that you ...

  7. Convert HTML Entities

    function convert(str) { // :) //return str; var HTML_Entities = { '&':'&', '<':'<', '& ...

  8. #import、#include以及@class的区别

    一.#import和#include的区别 当我们在代码中使用两次#include的时候会报错:因为#include相当于拷贝头文件中的声明内容,所以会报重复定义的错误 但是使用两次#import的话 ...

  9. 用nodej和glub-watcher写的监听go 项目自动编译,很鸡肋

    glub 一般都是很轻量的编译. go太重了,改一小个部分,就编译的话,多数是编译失败. 而且很消耗性能,还没想到完美的优化办法. 暂时用个定时器 监听2秒,停止1秒,如此循环,会减少些 “无效”的编 ...

  10. 统计学习方法 AdaBoost

    提升方法的基本思路 在概率近似正确(probably approximately correct,PAC)学习的框架中, 一个概念(一个类),如果存在一个多项式的学习算法能够学习它,并且正确率很高,那 ...