文章目录
一、量词(Quantifier)
???? 量词用以标记某个字符出现的次数
贪婪(Greedy) | 勉强(Reluctant) | 独占(Possessive) | 含义 |
---|---|---|---|
????{n} | ????{n}? | ????{n}+ | ???? 出现 n 次 |
????{n,m} | ????{n,m}? | ????{n,m}+ | ????出现 n 到 m 次 |
????{n,} | ????{,}? | ????{n,}+ | ????出现至少 n 次 |
????? | ?????? | ?????+ | ????出现 0 次或者 1 次 |
????* | ????*? | ????*+ | ????出现任意次 |
????+ | ????+? | ????++ | ????至少出现一次 |
???? 贪婪、勉强、独占的区别很大,后面介绍比较合适
二、字符串的 matches 方法底层
???? String 的 matches 方法底层用到了 Pattern 和 Matcher 类
public class TestDemo {
public static void main(String[] args) {
String input1 = "111庆222";
String input2 = "111庆";
String r = "1{3}庆";
// Pattern 的 matchers 匹配的是【整个】字符串
// match the entire region against the pattern.
// input1.matches(r): 看一看 input1 字符串是否完全匹配正则表达式 r
System.out.println(input1.matches(r)); // false
System.out.println(input2.matches(r)); // true
}
}
三、Matcher 类的其他方法
???? 上一节中了解了 Matcher 类的 matches 方法【如果整个 input(字符串)和正则表达式匹配,则返回 true】,这也是 String 的 matches 方法的底层
通过下面的代码引出 Matcher 类的其他方法:
public class TestDemo {
public static void main(String[] args) {
String input0 = "520";
String input1 = "888";
String input2 = "111_222_666";
String r = "\\d{3}";
System.out.println(input0.matches(r)); // true
System.out.println(input1.matches(r)); // true
// (1) 字符串的 matches 方法底层的 Matcher 类的 matches 方法是匹配整个字符串
// (2) 必须整个字符串和正则模式匹配(必须整个字符串和正则模式匹配)
// (3) input2 里面包含三个数字, 但不仅仅只是有三个数字, 无法和正则完全匹配
System.out.println(input2.matches(r)); // false
}
}
???? 上面代码中的 input2 无法完全匹配正则表达式 r(这是字符串底层 Matcher 类的 matches 方法的结果)
???? matches 方法是对整个字符串进行匹配,必须整个字符串匹配正则表达式
???? Matcher 类的 find 方法可以把字符串的部分序列和正则进行匹配,如果匹配成功,返回 true
(1) find、start、end、group
public class TestDemo {
public static void main(String[] args) {
String input = "111_222_666";
String r = "\\d{3}";
System.out.println(input.matches(r)); // false
// 假如正则表达式 r 语法错误, 会抛异常
Pattern pattern = Pattern.compile(r);
Matcher matcher = pattern.matcher(input);
// 如果调用 matches 方法, 则和 String 的 matches 方法返回的结果一样
System.out.println(matcher.matches()); // false
// input 中有子序列满足正则表达式 r 则返回 true
System.out.println(matcher.find()); // true
}
}
???? find: 如果从 input(给定字符串)中能够找到与 regex 匹配的子序列,则返回 true
???? 如果能够匹配成功,可以通过 start、end、group 方法获取更多的信息
???? 每次的查找范围会先剔除之前已经查找过的范围
???? start: 返回上一次匹配成功的开始索引
???? end: 返回上一次匹配成功的结束索引
???? group: 返回上一次匹配成功的子序列
public class TestDemo {
public static void main(String[] args) {
String input = "111_222_666";
String r = "\\d{3}";
Pattern pattern = Pattern.compile(r);
Matcher matcher = pattern.matcher(input);
boolean findResult = matcher.find();
System.out.println(findResult); // true
if (findResult) {
System.out.println("匹配的子序列: " + matcher.group());
System.out.println("起始索引: " + matcher.start());
System.out.println("结束索引: " + matcher.end());
}
}
}
(2) find 细节
???? 该方法(find)会从给定 input 的一开始(第一个字符)开始匹配
???? 但若该方法先前的调用是成功的,并且 matcher 还没有被重置,则从先前的匹配中还没有被匹配过的字符开始匹配
???? matcher 还没有被重置: matcher 是通过 input(给定字符串)创建的
???? 只要 input 没有改变, matcher 就没有被重置
???? find 方法被调用一次,就会在该 input 中匹配一次(从前往后匹配)
取出所有符合正则表达式的子序列:
public class TestDemo {
public static void main(String[] args) {
String input = "520_222_666";
String r = "\\d{3}";
Pattern pattern = Pattern.compile(r);
Matcher matcher = pattern.matcher(input);
// 当无法匹配到满足正则的子序列时, 结束循环
while (matcher.find()) {
System.out.println(matcher.group());
}
/*
520
222
666
*/
}
}
(3) 封装:查找字符串中匹配正则的子串
/**
* 返回给定字符串中所有成功匹配正则表达式的子串
*
* @param input 给定字符串
* @param regex 正则表达式
* @return 成功匹配正则表达式的子串 List
*/
public static List<String> okMatchRegexSubstrList(String input, String regex, int flags) {
if (input == null || regex == null) return null;
List<String> subStrings = new ArrayList<>();
Pattern p = Pattern.compile(regex, flags);
Matcher m = p.matcher(input);
while (m.find()) {
subStrings.add(m.group());
}
return subStrings;
}
}
四、贪婪、勉强、独占
???? 贪婪:
✏️ ① 先 “吞掉” 整个 input 进行匹配
✏️ ② 若匹配失败,则吐出最后一个字符,然后再次尝试匹配
✏️ ③ 重复该操作
public class TestDemo {
public static void main(String[] args) {
String regex = ".*good";
String input = "庆の6good8浩のgoodMorning";
List<String> ret = okMatchRegexSubstrList(input, regex, 0);
// 贪婪: [庆の6good8浩のgood]
printList("贪婪", ret, ", ");
}
/**
* 打印字符串 List
*/
public static void printList(String desc, List<String> list, String divider) {
int size = list.size();
if (size == 0) {
System.out.println("空数组");
return;
}
System.out.print(desc + ": [");
for (int i = 0; i < list.size(); i++) {
if (i != list.size() - 1) {
System.out.print(list.get(i) + divider);
}
System.out.print(list.get(i));
}
System.out.print("]");
}
public static List<String> okMatchRegexSubstrList(String input, String regex, int flags) {
if (input == null || regex == null) return null;
List<String> subStrings = new ArrayList<>();
Pattern p = Pattern.compile(regex, flags);
Matcher m = p.matcher(input);
while (m.find()) {
subStrings.add(m.group());
}
m.reset();
return subStrings;
}
}
???? 勉强:
✏️ ① 先 “吞掉” input 的第一个字符进行匹配
✏️ ② 若匹配失败,则再吞掉下一个字符,然后再次尝试匹配
✏️ ③ 重复该操作
public class TestDemo {
public static void main(String[] args) {
String regex = ".*?good";
String input = "庆の6good8浩のgoodMorning";
List<String> ret = okMatchRegexSubstrList(input, regex, 0);
// 勉强: [庆の6good, 庆の6good8浩のgood]
printList("勉强", ret, ", ");
}
???? 独占:
✏️ “吞掉” 整个 input 进行唯一的一次匹配(类似 equals 方法)
public class TestDemo {
public static void main(String[] args) {
String regex = ".*+good";
String input = "庆の6good8浩のgoodMorning";
List<String> ret = okMatchRegexSubstrList(input, regex, 0);
// 空数组
printList("独占", ret, ", ");
}
}
结束,如有错误!请赐教