复杂正则表达式原来是纸老虎

Home

最近在研究 HTML 解析,遇到一个正则表达式:

/([\w-:\*>]*)(?:\#([\w-]+)|\.([\w-]+))?(?:\[@?(!?[\w-:]+)(?:([!*^$]?=)["']?(.*?)["']?)?\])?([\/, ]+)/is

它用来匹配 CSS 选择器,比如 div > ul

过去见过很多这样复杂的表达式,我都本能地退缩了。今天就来彻底搞明白它!男人,该对自己狠一点!

匹配 div > ul

我找了一个网站 https://regex101.com/ ,能在线匹配,还有解释。

虽然有了右边的说明,清楚了一些。但还是不清楚具体匹配起来是怎样的。那就找几个例子,逐个分析。

具体出现这个正则表达式的代码是:

$matches = [];
preg_match_all($this->pattern, trim($selector).' ', $matches, PREG_SET_ORDER);

preg_match_all 的意思是获取所有满足模式的串。如果有:

preg_match_all("abc", "abcdabc", $matches)

第一个参数是模式,第二个参数是要匹配的字符串,第三个参数是结果引用。运行后,$matches 数组里会包含两个 abc

有了这个理解之后,上图的 div > ul 只匹配了前面四个字符 div >regex101 不支持 preg_match_all?还好加个叫 g 的修饰符就行了:

加了 g 后,会匹配所有的,而不是匹配到第一个就返回。

加了之后,我们匹配到了 div > ul

右边显示,第一个匹配中,即 div,我们用第一组的规则匹配到了 div,然后用第七组的规则匹配到了空格 ` `。

我们接着来看第一组规则的解释:

在这一大串表达式中,第一个括号括起来的叫第一组规则。这是一个捕获组。括号自身不匹配,而是用来分组。[] 表示一个字符集合,里面的规则说明这是一个怎样的字符集合。这个字符集合里有:

[\w-:\*>]* 最后的 * 表示前面的字符能出现 0 次或无数次,但要匹配尽可能多的次数。之所以能匹配 div 是因为,\w 匹配了 div。之所以不再继续匹配后面的空格,是因为空格没有出现在 [] 中。捕获组的意思是,这组匹配会出现在结果数组中。相对应的还有非捕获组,语法是 (?:)。上面的 ([\w-:\*>]*) 如果不需要这组结果,可以记为 (?:[\w-:\*>]*)

那么不出现在结果中,直接不用括号不就行了?括号是为了分组,分组还是很有意义的。可以参考 《What is a non capturing group? (?:) - StackOverflow》

接着讲完了 div 满足第一组规则后,讲下空格 ` ` 为什么满足第七组的规则。

[\/, ] 的意思是匹配这四个字符的任意一个,+ 表示前面的匹配出现一次或无数次,次数要尽可能多。所以因为这四个字符包含空格,就匹配了我们的空格。又因为 div 之后下一个字符是 >,所以不再满足第七组的规则,不继续匹配了。

搞明白了 div 的匹配。那为什么第二到第六组的规则没有匹配这里的空格,而是留给了第七组?

第二部分的解释:

首先 (?:) 表示这是一个非捕获组。最后面的 ? 表示前面的匹配可以出现 0 次或 1 次。所以上面的 (?:\#([\w-]+)|\.([\w-]+))? 可以有或没有。去掉外层的修饰符后,剩下的是 \#([\w-]+)|\.([\w-]+),中间的 | 表示或,满足其中一个即可。\#([\w-]+) 中的 \# 匹配 # 字符,[\w-]+ 匹配其他字符。再看后一半,\.([\w-]+) 中的 . 匹配 . 字符。

所以 2 到 6 组都可能因为空格不是这些组要求的开头字符而不满足,又因为这些组有个 ? 修饰符,不满足也可以,因此跳到了第七组。

接着 div > ul 后面的 >,还是一样:

第一组规则 ([\w-:\*>]*) 匹配了 >,第七组规则 ([\/, ]+) 匹配了空格。接着 uldiv 一样。

匹配 #answer-4185009 > table > tbody > td.answercell > div > pre

接着来一个稍微复杂一点的选择器 #answer-4185009 > table > tbody > td.answercell > div > pre(你也可以打开 https://regex101.com/ 把这个粘贴到那里去来测试):

这是从 Chrome 里复制粘贴的:

第一个匹配:

因为第一组的规则 ([\w-:\*>]*)[] 里的字符集没有一个能匹配 #,接着因为最后面的 * 支持匹配 0 次或无数次,这里是 0 次。接着第二组规则的描述是:

上面已经分析过。直接来看 | 前的 \#([\w-]+)\## 匹配了,[\w-]+ 匹配了 answer-4185009。后面的 \.([\w-]+),如果是 .answer-4185009 就会应用这个匹配。

接着来看 td.answercell 这个匹配,

第一组的规则 ([\w-:\*>]*) 匹配了 td,第二大部分的 (?:\#([\w-]+)|\.([\w-]+))? 的后面部分,即 \.([\w-]+),匹配了 .answercell

这个选择器的分析也到此结束。

匹配 a[href="http://google.com/"]

接着我们来匹配选择器 a[href="http://google.com/"]

看看第三大块:

第三大块的表达式为 (?:\[@?(!?[\w-:]+)(?:([!*^$]?=)["']?(.*?)["']?)?\])?,首先最外层 (?:) 表示这是一个非捕获组,最后面的 ? 表示这整个大的可以匹配 0 次或 1 次。去掉之后为 \[@?(!?[\w-:]+)(?:([!*^$]?=)["']?(.*?)["']?)?\]\[ 匹配 [ 字符。@? 表示 @ 字符可有可无。那么接下来的一组 (!?[\w-:]+)! 可有可无,[\w-:]+ 匹配 href。接着的一组 (?:([!*^$]?=)["']?(.*?)["']?) 是非捕获组,去掉最外层后是 ([!*^$]?=)["']?(.*?)["']?。这里的 ([!*^$]?=)[!*^$]? 表示匹配 0 个或 1 个 [] 里的字符。接着 = 直接匹配。接着 ["']?(.*?)["']?,匹配 "http://google.com/"["']? 表示匹配 "' 或者两个都不匹配,去掉这个最外层后是 (.*?) 匹配 http://google.com/,这里的 *? 表示尽可能少的匹配,也就是说,如果有 "',要留给后面的表达式 ["']? 来匹配。所以不至于匹配到 http://google.com/",而是只匹配 http://google.com/。所以整个选择器 `a[href=”http://google.com/

”]` 就匹配结束了。

总结

终于搞懂了!再让我们理清一次,首先整个复杂的表达式 ([\w-:\*>]*)(?:\#([\w-]+)|\.([\w-]+))?(?:\[@?(!?[\w-:]+)(?:([!*^$]?=)["']?(.*?)["']?)?\])?([\/, ]+) 由四大部分组成:

最复杂的第三部分又由这几个部分组成:

所以这些足够小的部分都可以逐个击破。然后找多一点例子,看看每个例子是怎么匹配的,同时结合 https://regex101.com/ 的解释来分析。就把这个看似复杂的正则表达式搞懂了,原来它是个纸老虎!


Back Donate