在Java中,正则表达式中的捕获组(Capturing Groups)和非捕获组(Non-Capturing Groups)是用来对匹配的部分进行分组的技术。它们在处理复杂的匹配模式时非常有用。
捕获组 (Capturing Groups)
捕获组不仅将一部分正则表达式分组,还“捕获”了该组匹配的文本。这允许你在后续的代码中引用这些匹配的子串。捕获组通过圆括号 ()
来定义。
示例
假设我们有以下字符串:
String text = "Hello, my name is John Doe.";
如果我们想提取名字部分,可以使用这样的正则表达式:
String regex = "my name is (\\w+ \\w+)";
Pattern pattern = Pattern.compile(regex);
Matcher matcher = pattern.matcher(text);
if (matcher.find()) {
System.out.println("Full Name: " + matcher.group(1)); // 输出 "John Doe"
}
在这个例子中,(\\w+ \\w+)
是一个捕获组,它匹配两个由空格分隔的单词,并且可以通过 matcher.group(1)
来访问这个匹配的子串。
非捕获组 (Non-Capturing Groups)
非捕获组同样使用圆括号来定义,但它们不会捕获匹配的文本,因此不能通过 group()
方法来访问。非捕获组通常用于逻辑分组,但不需要保留匹配的结果。非捕获组通过 (?:...)
来定义。
示例
假设我们有以下字符串:
String text = "Hello, my name is John Doe and I am 30 years old.";
如果我们只想匹配包含年龄的句子,但不关心具体的名字,可以使用非捕获组:
String regex = "my name is (?:\\w+ \\w+) and I am (\\d+) years old";
Pattern pattern = Pattern.compile(regex);
Matcher matcher = pattern.matcher(text);
if (matcher.find()) {
System.out.println("Age: " + matcher.group(1)); // 输出 "30"
}
在这个例子中,(?:\\w+ \\w+)
是一个非捕获组,它只用于逻辑分组,而不会捕获任何文本。因此,matcher.group(1)
返回的是第一个真正的捕获组,即年龄部分。
总结
- 捕获组 (
(...)
): 用于分组并捕获匹配的文本,可以通过group()
方法访问。 - 非捕获组 (
(?:...)
): 仅用于逻辑分组,不捕获匹配的文本,不能通过group()
方法访问。
选择使用捕获组还是非捕获组取决于你是否需要在后续处理中引用匹配的子串。如果不需要引用,使用非捕获组可以提高性能,因为JVM不需要存储这些匹配的子串。