深入入门正则表达式(java)
一.入门基础
1.元字符
很多人对正则表达式的印象就是乱码。。许许多多的符号组合在一起,偶见单词,正则确实是这样的,所以下面我们要看看这些符号都是什么意思
有些符号不是大家看到的字面上的意思:比如“.”、“!”、“?” ……
这些符号就称之为元字符
下面我们逐一说明
“\” :转义符号,在字符组内依然是元字符。
在检查一行文本时
“^” :脱字符:表示一行的开始
“$” :美元符号:表示一行的结束
字符组
“[]” :一对中括号,里面可以添加任何内容,比如[hate],括号里面的内容是或者的关系,这个正则的意义是:我想匹配一个字符,这个字符可以是h,可以是a,也可以是t或e。
记住:字符组最终只会匹配一个字符。 即使你的目标字符串是hate,那么一次匹配成功的结果也只是第一个字母h,并不是整个单词,如果我就想匹配hate这个单词怎么办?很容易,正则内容为“hate”,在字符组外面的字符的关系是“和,并且”的意思。
注意:字符组内的元字符和字符组外的元字符并不一样,即使字符一样,表示的意义也未必相同 (*)
我们先学习下一个内容,然后再来给大家解释上面的这句话
“[^]” :括号里面以“^”开头,是字符组的否定形式 ,比如:[^hate]表示:匹配除了h、a、t、e以外的其他字符,依然只会匹配一个字符
之前刚刚学过“^”表示一行的开始,但是脱字符位于[]内的起始位置则表示否定,这也解释了(*) 的内容
如果脱字符位于括号内其他位置表示什么呢?那它表示它自己,此时并不再是一个元字符
“-” :减号,可以在字符组内表示区间、范围。比如“[a-z]”,表示匹配a到z直接的任意一个小写字母,如果是“[-z]”,那么“-”只表示它自己,没有任何特殊意义,它在字符组外也没有任何特殊意义。
ps:关于“^”、“$”、“-”的其他用法将在高级基础篇讲述
“.” :点号,匹配一个任意字符的字符组简便写法。“.”在字符组内没不是元字符
ps:“.”也不是万能的,有些字符无法匹配,之后会详细解释并给出替代方案
“|” :竖线,表示或者,这个很容易记忆,如果想匹配hate或者like,那么正则为:“hate|like”
注意:如果想匹配I hate you和I like you。那么正则为:“I (hate|like) you”。如果写成“I hate|like you”,那么匹配的结果为“I hate”或者是“like you”了
这里圆括号的作用是控制竖线的作用范围,“()”有很多作用,稍后再说
“\b” :它的作用是单词分隔符,如果我想匹配like,那么正则为“like”,没错,但是会得到一些我不想要的结果,比如likely也会跑到结果集中,我可不想要这些单词。那么修改正则为:“\blike\b”,这回就只能匹配like这个单词了。
注意:java中的单词分隔符为“\b”,有些语言的单词分隔符为“\<” 和“\>” 。
单词边界是什么呢?其实正则没有那么聪明,它不会识别“Ilikeyou”为“I like you”,它只是找到数字和字母的起始位置和结束位置而已
“\w” :单词字符。在java中相当于“[a-zA-Z0-9_]”。 但是java中的“\b”却支持Unicode字符。
下面我们来看看正则中的“数字” - 量词
“?” :表示0个至1个
“*” :表示0个至任意多个
“+” :表示至少一个
“{min,max}” :区间量词。“{2,5}”,表示匹配2到5次。“{2,}”表示至少两次,“{2}”表示只匹配两次。 “{,2}”,正则错误,无意义
举个例子:正如上面的反面教程所说,如果想匹配一个正整数,那么应该如何来做
首先我们需要明确:我不想匹配0123,只想匹配123这样的数字,0不能开头,第二个要求就是除了数字不能有其他字符了
之前我们学过字符组了,“[0-9]”可以表示0至9中任意一个数字,那么根据上面的量词,我们改正则为“[0-9]+”,表示至少有一个数字,至多没有限制。但是0123这样的数字也会满足要求。那么我们再修改一下,首先第一位不能为0,那么这一位可以写为“[1-9]”,表示匹配1到9的任何一个数字,之后呢?后面的数字有没有都无所谓了,有,那就是多位数,没有,那就是一位数。所以正则为“[1-9][0-9]*”。
常见字符范围缩写
贪婪与懒惰
我们再来看一个量词的例子。比如我想匹配一个单词,正则可以这么写“[a-zA-Z]+”,或者“\w+”
/**
* 测试 正则表达式
* @ClassName: regexTest_1
* @Description:
* @author xingle
* @date 2014-4-21 上午11:15:09
*/
public class regexTest_1 {
public static void main(String[] args) {
String input = "there is a book on the desk.";
// 正则表达式
String regex = "\\w+";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(input);
boolean found = false;
while (m.find()) {
found = true;
System.out.println("输入:"+input);
System.out.printf("匹配的字符: \"%s\" 索引开始为 %d and 结束索引为 %d.%n",
m.group(), m.start(), m.end());
System.out.println();
}
if(found==false){
System.out.println("没有找到匹配字符串!");
} } }
执行结果:
输入:there is a book on the desk.
匹配的字符: "there" 索引开始为 0 and 结束索引为 5.
输入:there is a book on the desk.
匹配的字符: "is" 索引开始为 6 and 结束索引为 8.
输入:there is a book on the desk.
匹配的字符: "a" 索引开始为 9 and 结束索引为 10.
输入:there is a book on the desk.
匹配的字符: "book" 索引开始为 11 and 结束索引为 15.
输入:there is a book on the desk.
匹配的字符: "on" 索引开始为 16 and 结束索引为 18.
输入:there is a book on the desk.
匹配的字符: "the" 索引开始为 19 and 结束索引为 22.
输入:there is a book on the desk.
匹配的字符: "desk" 索引开始为 23 and 结束索引为 27.
有个问题就出来了:“\w+”表示至少一个“\w”,那么为什么结果不是“t”、“h”、“e”、“r”、“e”,而是“there”。
上面的量词,除了“{times}”这种指定匹配次数的,其余默认均为贪婪匹配。也就是说尽可能多的匹配。
相对的就有惰性匹配,那么惰性匹配如何使用?
下面修改一下例子:“\w*e”表示以e结尾的单词,现在这里的*还是贪婪匹配。
public class regexTest_1 {
public static void main(String[] args) {
String input = "there is a book on the desk.";
// 正则表达式
String regex = "\\w*e";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(input);
int c = 0;
boolean found = false;
System.out.println("输入:"+input);
while (m.find()) {
c++;
found = true;
System.out.println("第 "+c+"个");
System.out.printf("匹配的字符: \"%s\" 索引开始为 %d and 结束索引为 %d.%n",
m.group(), m.start(), m.end());
System.out.println();
}
if(found==false){
System.out.println("没有找到匹配字符串!");
}
}
}
执行结果:
输入:there is a book on the desk.
第 1个
匹配的字符: "there" 索引开始为 0 and 结束索引为 5.
第 2个
匹配的字符: "the" 索引开始为 19 and 结束索引为 22.
第 3个
匹配的字符: "de" 索引开始为 23 and 结束索引为 25.
如果我想匹配到单词中的第一个e,那么如何修改呢?
public class regexTest_1 {
public static void main(String[] args) {
String input = "there is a book on the desk.";
// 正则表达式 匹配到每个单词的第一个e结束
String regex = "\\b\\w*?e";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(input);
int c = 0;
boolean found = false;
System.out.println("输入:"+input);
while (m.find()) {
c++;
found = true;
System.out.println("第 "+c+"个");
System.out.printf("匹配的字符: \"%s\" 索引开始为 %d and 结束索引为 %d.%n",
m.group(), m.start(), m.end());
System.out.println();
}
if(found==false){
System.out.println("没有找到匹配字符串!");
}
}
}
执行结果:
输入:there is a book on the desk.
第 1个
匹配的字符: "the" 索引开始为 0 and 结束索引为 3.
第 2个
匹配的字符: "the" 索引开始为 19 and 结束索引为 22.
第 3个
匹配的字符: "de" 索引开始为 23 and 结束索引为 25.
还是来看there,这回“\w+”只匹配了“th”,并没有匹配到“ther”才停止。
惰性匹配就是尽可能少的匹配,使用方法就是在量词后面加上“?”
如果量词后面没有“?”等其他量词,那么就是默认的贪婪匹配。
要是匹配整个单词,结尾加单词分隔符“\b”
public class regexTest_1 {
public static void main(String[] args) {
String input = "there is a book on the desk.";
// 正则表达式 匹配到以e结束的整个单词
String regex = "\\b\\w*?e\\b";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(input);
int c = 0;
boolean found = false;
System.out.println("输入:"+input);
while (m.find()) {
c++;
found = true;
System.out.println("第 "+c+"个");
System.out.printf("匹配的字符: \"%s\" 索引开始为 %d and 结束索引为 %d.%n",
m.group(), m.start(), m.end());
System.out.println();
}
if(found==false){
System.out.println("没有找到匹配字符串!");
}
}
}
执行结果:
输入:there is a book on the desk.
第 1个
匹配的字符: "there" 索引开始为 0 and 结束索引为 5.
第 2个
匹配的字符: "the" 索引开始为 19 and 结束索引为 22.
“?”,“*”,“+”:也叫匹配优先量词
“*?”,“+?”,“??”:也叫忽略优先量词
其实还有一种量词:
“?+”,“*+”,“++”:占有优先量词 。 (支持这种量词的正则引擎很少,java支持)
这节不讨论这种类型量词,之后的章节讨论
() :将括号里面的内容作为一个独立的单元,理解为一组。
捕获组(capturing group)是将多个字符作为单独的单元来对待的一种方式。构建它们可以通过把字符放在一对圆括号中而成为一组。例如,正则表达式(dog)
建了单个的组,包括字符“d”“o”和“g”。匹配捕获组输入的字符串部分将会存放于内存中,稍后通过反向引用再次调用。
在 Pattern 的 API 描述中,捕获组通过从左至右计算开始的圆括号进行编号。例如,在表达式
((A)(B(C)))
中,有下面的四组:1.
((A)(B(C)))
2.
(A)
3.
(B(C))
4.
(C)
要找出当前的表达式中有多少组,通过调用 Matcher 对象的 groupCount 方法。groupCount 方法返回 int 类型值,表示当前 Matcher 模式中捕获组的数量。例如,groupCount 返回 4 时,表示模式中包含有 4 个捕获组。
有一个特别的组——组 0,它表示整个表达式。这个组不包括在 groupCount 的报告范围内。以
(?
开始的组是纯粹的非捕获组(non-capturing group),它不捕获文本,也不作为组总数而计数Matcher 中的一些方法,可以指定 int 类型的特定组号作为参数,因此理解组是如何编号的是尤为重要的。
public int start(int group):返回之前的匹配操作期间,给定组所捕获的子序列的初始索引。
public int end(int group):返回之前的匹配操作期间,给定组所捕获子序列的最后字符索引加 1。
public String group (int group):返回之前的匹配操作期间,通过给定组而捕获的输入子序列。
“(?:)” :非捕获型括号。“(?:desk)”会匹配到desk这个字符串,但是如果你企图使用反向引用“(?:desk) \1”,那么就会出现错误。这样的操作是非法的。
注意:非捕获型括号不影响捕获计数
好处:
1.提高效率,很容易理解,不记住捕获的内容也就不占用内存了
2.结构清晰
1 概述
捕获组捕获到的内容,不仅可以在正则表达式外部通过程序进行引用,也可以在正则表达式内部进行引用,这种引用方式就是反向引用。
反向引用的作用通常是用来查找或限定重复、查找或限定指定标识配对出现等等。
对于普通捕获组和命名捕获组的引用,语法如下:
普通捕获组反向引用:\k<number>,通常简写为\number
命名捕获组反向引用:\k<name>或者\k'name'
普通捕获组反向引用中number是十进制的数字,即捕获组的编号;命名捕获组反向引用中的name为命名捕获组的组名。
2 反向引用匹配原理
捕获组(Expression)在匹配成功时,会将子表达式匹配到的内容,保存到内存中一个以数字编号的组里,可以简单的认为是对一个局部变量进行了赋值,这时就可以通过反向引用方式,引用这个局部变量的值。一个捕获组(Expression)在匹配成功之前,它的内容可以是不确定的,一旦匹配成功,它的内容就确定了,反向引用的内容也就是确定的了。
反向引用必然要与捕获组一同使用的,如果没有捕获组,而使用了反向引用的语法,不同语言的处理方式不一致,有的语言会抛异常,有的语言会当作普通的转义处理。
2.1 从一个简单例子说起
源字符串:abcdebbcde
正则表达式:([ab])\1
对于正则表达式“([ab])\1”,捕获组中的子表达式“[ab]”虽然可以匹配“a”或者“b”,但是捕获组一旦匹配成功,反向引用的内容也就确定了。如果捕获组匹配到“a”,那么反向引用也就只能匹配“a”,同理,如果捕获组匹配到的是“b”,那么反向引用也就只能匹配“b”。由于后面反向引用“\1”的限制,要求必须是两个相同的字符,在这里也就是“aa”或者“bb”才能匹配成功。
考察一下这个正则表达式的匹配过程,在位置0处,由“([ab])”匹配“a”成功,将捕获的内容保存在编号为1的组中,然后把控制权交给“\1”,由于此时捕获组已记录了捕获内容为“a”,“\1”也就确定只有匹配到“a”才能匹配成功,这里显然不满足,“\1”匹配失败,由于没有可供回溯的状态,整个表达式在位置0处匹配失败。
正则引擎向前传动,在位置5之前,“([ab])”一直匹配失败。传动到位置5处时,,“([ab])”匹配到“b”,匹配成功,将捕获的内容保存在编号为1的组中,然后把控制权交给“\1”,由于此时捕获组已记录了捕获内容为“b”,“\1”也就确定只有匹配到“b”才能匹配成功,满足条件,“\1”匹配成功,整个表达式匹配成功,匹配结果为“bb”,匹配开始位置为5,结束位置为7。
扩展一下,正则表达式“([a-z])\1{2}”也就表达连续三个相同的小写字母。
常用空白字符
“\s” :表示所有空白字符。
“\t” :制表符。
“\n” :换行符。
“\r” :回车。
一些其他常用的缩略表示
“\S” :除“\s”之外的任何字符
“\w” :等同于[a-zA-Z0-9_]
“\W ” :除“\w”之外的任何字符
“\d” :等同于[0-9]
“\D” :除“d”之外的任何字符
有些工具不支持,比如EditPlus v3.10 中的查找就不支持\d等。
环视(零宽断言)
环视分为顺序和逆序,肯定和否定,组合一下一共4种。下面就看看环视到底是什么
“(?=)” :顺序肯定环视:(从左至右)查看文本,如果能够匹配,就返回匹配成功信息。
“(?<=)” :逆序肯定环视:(从右至左)查看文本,如果能够匹配,就返回匹配成功信息。
“(?!)” :顺序否定环视:(从左至右)查看文本,如果不能够匹配,就返回匹配成功信息。
“(?<!)” :逆序否定环视:(从右至左)查看文本,如果不能够匹配,就返回匹配成功信息。
下面看几个简单的实例
例:下面有两句话,加入你只想找到book,不想找到books
there is a book on the desk.
there are some books on the desk.
最简单的办法是:“book\b”,这很容易理解,book后面跟着单词分隔符,book后面如果是s,那么肯定被认为是一个单词,所以这样不会匹配到books。如果用环视,应该如何书写呢
“book(?!\w)”
public class regexTest_1 {
public static void main(String[] args) {
String input = "there is a book on the desk. there are some books on the desk.";
// 正则表达式 等价于"book\\b"
String regex = "book(?!\\w)";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(input);
int c = 0;
boolean found = false;
System.out.println("输入:"+input);
while (m.find()) {
c++;
found = true;
System.out.println("第 "+c+"个");
System.out.printf("匹配的字符: \"%s\" 索引开始为 %d and 结束索引为 %d.%n",
m.group(), m.start(), m.end());
System.out.println();
}
if(found==false){
System.out.println("没有找到匹配字符串!");
}
}
}
正则中的book很好理解,依次匹配b、o、o、k,然后呢,\w在上面说过等同于[a-zA-Z0-9],“(?!\w)”是说:我要找这样一个位置,这个位置的后面不能是\w。
第一句话中,在匹配了book后,发现紧跟的是一个空格,恩,不是\w中的内容,匹配成功。
如果想匹配books的book怎么办,很简单“book(?=s)”
注意:环视不会占用字符!环视查找的是字符与字符之间的位置。
环视括号并没有捕获字符的功效,所以不能使用反向引用。
参考资料:
深入入门正则表达式(java)的更多相关文章
- 学Android开发,入门语言java知识点
学Android开发,入门语言java知识点 Android是一种以Linux为基础的开源码操作系统,主要使用于便携设备,而linux是用c语言和少量汇编语言写成的,如果你想研究Android,就去学 ...
- Thrift入门及Java实例演示<转载备用>
Thrift入门及Java实例演示 作者: Michael 日期: 年 月 日 •概述 •下载配置 •基本概念 .数据类型 .服务端编码基本步骤 .客户端编码基本步骤 .数据传输协议 •实例演示(ja ...
- Java开发者必备的10大学习网站,送给入门学习java的你,请收下!
作为开发者来说,必备的除了对编码的热情还要有自己的一套技巧,另外不可缺少的就是平时学习的网站.以下本人收集的 Java 开发者必备的网站,这些网站可以提供信息.以及一些很棒的讲座 , 还能解答一般问题 ...
- Java入门:Java下载与安装方法
本文适合刚入门的Java编程的初学者阅读. JDK有两种下载方法,一个是官网下载,另一个是第三方网站下载.官网速度也许有点慢,慢的话可以考虑去第三方网站下载. 一.官网下载 1. 访问地址:http: ...
- ElasticSearch入门-搜索(java api)
ElasticSearch入门-搜索(java api) package com.qlyd.searchhelper; import java.util.Map; import net.sf.json ...
- Java入门——初识Java
Java入门——初识Java 摘要:本文主要对Java这门编程语言进行简单的介绍. Java简介 说明 Java语言历时十多年,已发展成为人类计算机史上影响深远的编程语言,从某种程度上来看,它甚至超出 ...
- # 095 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 03 封装总结 01 封装知识点总结
095 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 ...
- 094 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 02 static关键字 04 static关键字(续)
094 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 ...
- 093 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 02 static关键字 03 static关键字(下)
093 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 ...
- 092 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 02 static关键字 02 static关键字(中)
092 01 Android 零基础入门 02 Java面向对象 02 Java封装 01 封装的实现 03 # 088 01 Android 零基础入门 02 Java面向对象 02 Java封装 ...
随机推荐
- 目前比较全的CSS重设(reset)方法总结
在当今网页设计/开发实践中,使用CSS来为语义化的(X)HTML标记添加样式风格是 重要的关键.在设计师们的梦想中都存在着这样的一个完美世界:所有的浏览器都能够理解和适用多有CSS规则,并且呈现相同的 ...
- Codeforces Round #250 (Div. 2)A(英语学习)
链接:http://codeforces.com/contest/437/problem/A A. The Child and Homework time limit per test 1 secon ...
- Codeforces Round #281 (Div. 2) C. Vasya and Basketball 二分
C. Vasya and Basketball time limit per test 2 seconds memory limit per test 256 megabytes input stan ...
- LTE Module User Documentation(翻译3)——仿真输出
LTE用户文档 (如有不当的地方,欢迎指正!) 6 仿真输出 ns-3 LTE 模型当前支持输出 PHY, MAC, RLC 和 PDCP 级别的 Key Performance Indicators ...
- iOS - OC NSRange 范围
前言 结构体,这个结构体用来表示事物的一个范围,通常是字符串里的字符范围或者集合里的元素范围. typedef struct _NSRange { NSUInteger location; // 表示 ...
- 超实用压力测试工具-ab工具
在学习ab工具之前,我们需了解几个关于压力测试的概念 吞吐率(Requests per second)概念:服务器并发处理能力的量化描述,单位是reqs/s,指的是某个并发用户数下单位时间内处理的请求 ...
- 【转】分析Linux和windows动态库
原文地址:http://www.cnblogs.com/chio/archive/2008/11/13/1333119.html 摘要:动态链接库技术实现和设计程序常用的技术,在Windows和Lin ...
- python的最最最最最基本语法(2)
函数的定义: 使用def语句,依次写出函数名.括号.括号中的参数和冒号:,然后,在缩进块中编写函数体,函数的返回值用return语句返回. 当用return 返回多个值时,返回的其实是一个tuple, ...
- jQuery学习笔记:attr()与prop()的区别
先看看官方文档是如何解释两者之间功能差异的: attr() Get the value of an attribute for the first element in the set of matc ...
- html5移动端制作知识点总结
一.测试工具:1.Chrome 2.Opera Mobile二.分辨率:一般现代手机最小320px,最大640px.三.全屏流体设计: 1.腾讯新闻:http://xw.qq.com/ 2.途牛旅游: ...