Home >Backend Development >PHP Tutorial >Capturing groups and non-capturing groups of PHP regular expressions, php regular expressions_PHP tutorial

Capturing groups and non-capturing groups of PHP regular expressions, php regular expressions_PHP tutorial

WBOY
WBOYOriginal
2016-07-12 09:05:25912browse

Capturing groups and non-capturing groups of PHP regular expressions, PHP regular expressions

I encountered a regular matching problem today, and suddenly I came across the concept of capturing groups, manual I also skimmed over the above. When I turned to Baidu, I accidentally found that there are special uses of regular capture groups in C# and Java. When I searched for the keyword PHP, there was no relevant content. I tried it myself and found that it is also feasible in PHP, so To sum up, while sharing, I also hope that some experts and careful learners can find the problems in my understanding.

What is a capturing group

Capturing group syntax:

Character

Description

Example

(pattern)

Match the pattern and capture the results, automatically setting the group number.

(abc) d

Match abcd or abcabcd

(?420ae5b27bd6f6b72458bb91a31cff15pattern)

or

(?'name'pattern)

Match pattern and capture the result, set name to the group name.

num

Backreference to a capturing group. where num is a positive integer.

(w)(w)21

Matches abba

k1971065f98ee2d1f9f23a7d9551084fe

or

k' name '

Backreference to a named capturing group. where name is the capture group name.

(?1cd70caa06cc6f4a046ab31342a5e7b4w)abck1cd70caa06cc6f4a046ab31342a5e7b4

Match xabcx

我们先看一下PHP的正则匹配函数

int preg_match ( string $pattern , string $subject [, array &$matches [, int $flags = 0 [, int $offset = 0 ]]] )

前面两项是我们常用的,$pattern是正则匹配模式,$string是要匹配的字符串。

array &$match,它是一个数组,&表示匹配出来的结果会被写入$match中。

int $flags 如果传递了这个标记, 对于每一个出现的匹配返回时会附加字符串偏移量(相对于目标字符串的)。

int $offset 用于指定从目标字符串的某个未知开始搜索(单位是字节)。

我们主要看一下$match的值里会有什么:

$mode = '/a=(\d+)b=(\d+)c=(\d+)/';
$str='**a=4b=98c=56**';
$res=preg_match($mode,$str,$match);
var_dump($match);

结果如下:

array (size=4)
  0 => string 'a=4b=98c=56' (length=11)
  1 => string '4' (length=1)
  2 => string '98' (length=2)
  3 => string '56' (length=2)

现在我们知道了什么是捕获组,捕获组是正则表达示中以()括起来的部分,每一对()是一个捕获组。

PHP会为它编号,从1开始。至于为什么会从1开始,那是因为PHP把匹配到的完整字符串编号为0。

如果有多个括号或嵌套括号,按左边括号出现的顺序来进行编号,如图:

按图中的匹配模式匹配时,捕获组的123号分别是红绿蓝。

捕获组的忽略与命名

我们还可以阻止PHP为匹配组的编号:在匹配组中模式前加  ?:

$mode = '/a=(\d+)b=(?:\d+)c=(\d+)/';

这样,匹配结果就会变成:

array (size=3)
 0 => string 'a=4b=98c=56' (length=11)
 1 => string '4' (length=1)
 2 => string '56' (length=2)

当然,我们也可以在括号的内部为它给它独特的名字。

命名子组可以接受(?8a11bc632ea32a57b3e3693c7987c420), (?'name') 以及(?P8a11bc632ea32a57b3e3693c7987c420)语法. 之前版本仅接受(?P8a11bc632ea32a57b3e3693c7987c420)语法.

例如:$mode = '/a=(\d+)b=(?P6fd21b5631009a6df3e3d5bae31abfd3\d+)c=(\d+)/';

使用时结果为:

array (size=5)
 0 => string 'a=4b=98c=56' (length=11)
 1 => string '4' (length=1)
 'sec' => string '98' (length=2)
 2 => string '98' (length=2)
 3 => string '56' (length=2)

在保留索引数组的同时,加上一个关联项,key值为捕获组名。

捕获组的反向引用

我们在用preg_replace()函数进行正则替换时,我们还可以使用 \n 或 $n 来引用第n个捕获组.

$mode = '/a=(\d+)b=(\d+)c=(\d+)/';
$str='**a=4b=98c=56**';
$rp='\1/$2/\3/';
echo preg_replace($mode,$rp,$str);//**4/98/56/**

\1表示捕获组1(4),$2为捕获组2(98),\3为捕获组3(56)。

非捕获组的用法:

非捕获组语法:

字符 

描述

Example

(?:pattern)

Matches pattern, but does not capture the matching result.

'industr(?:y|ies)

Match 'industry' or 'industries'.

(?=pattern)

Zero-width forward lookup, does not capture matching results.

'Windows (?=95|98|NT|2000)'

Matches "Windows" in "Windows2000"

Does not match "Windows" in "Windows3.1".

(?!pattern)

Zero-width negative lookup, does not capture matching results.

'Windows (?!95|98|NT|2000)'

Matches "Windows" in "Windows3.1"

Does not match "Windows" in "Windows2000".

(?<=pattern)

Zero-width forward lookback, no matches captured.

'2000 (?<=Office|Word|Excel)'

Match "2000" in "Office2000"

Does not match "2000" in "Windows2000".

(?pattern)

零宽度负向回查,不捕获匹配结果。

'2000 (?

匹配 " Windows2000" 中的 "2000"

不匹配 " Office2000" 中的 "2000"。

为什么称为非捕获组呢?那是因为它们有捕获组的特性,在匹配模式的()中,但是匹配时,PHP不会为它们编组,它们只会影响匹配结果,并不作为结果输出。

/d(?=xxx)    匹配"后面是xxx的一个数字"。

注意格式:只能放在匹配模式字符串之后!

例如:

$pattern='/\d(&#63;=abc)/';
$str="ab36abc8eg";
$res=preg_match($pattern,$str,$match);
var_dump($match);//6

匹配的6,因为只有它作为一个数字,后面还有abc。

(?<=xxx) /d 匹配"前面是xxx的一个数字"

注意格式:只能放在匹配模式字符串之前!

例如:

$pattern='/(&#63;<=abc)\d/';
$str="ab36abc8eg";
$res=preg_match($pattern,$str,$match);
var_dump($match);//8

匹配的8,因为只有它作为一个数字,后面还有abc。

与(?=xxx)  (?<=xxx)相对的是(?!=xxx)  (?

它表示前面/后面不是xxx的字符串,这里就不再举例了。

如果您觉得本博文对您有帮助,您可以推荐或关注我,如果您有什么问题,可以在下方留言讨论,谢谢。

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/1068823.htmlTechArticlePHP正则表达式之捕获组与非捕获组,php正则表达式 今天遇到一个正则匹配的问题,忽然翻到有捕获组的概念,手册上也是一略而过,百度时...
Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn