PHP多行匹配preg_match_all

4

我有以下代码:

$content = <<<HTML
<p>{wrapper btnlabel=&quot;AAA&quot;}zzzzzzzzzzzzzzzzzzzz{/wrapper}</p>

<p>{wrapper btnlabel=&quot;Text&quot;}</p>

<table>
<tbody>
<tr>
<th scope="row">123</th>
<td>1 123</td>
<td>12 123</td>
<td>3 123</td>
<td>1 123 123</td>
</tr>
<tr>
<th scope="row">123</th>
<td>2700 123.</td>
<td>1800 123.</td>
<td>1000 123.</td>
<td>300 123.</td>
</tr>
</tbody>
</table>

<p>{/wrapper}</p>
HTML;

preg_match_all('#(?>{wrapper btnlabel=(?>"|&quot;)(.+)(?>"|&quot;)})(.*){/wrapper}#', $content, $matches);
var_dump($matches);

模式与第二个 {wrapper...} 不匹配。我认为这是因为它被分成多行了。但是当我尝试使用 s 修饰符时,它会将第一个 {wrapper} 到最后一个 {/wrapper} 的整个内容匹配。将 \n 替换为 '' 没有帮助。所以我很困惑。也许我漏掉了什么?https://ideone.com/ZMpSJ7 - 这里是相同的测试代码。

4
请看这里: $re = "/(?>{wrapper btnlabel=(?>\"|&quot;)(.+?)(?>\"|&quot;)})(.*?){\\/wrapper}/s";。问题出在贪婪匹配的 (.+)(.*) 上面。 - Wiktor Stribiżew
现在可以工作了!谢谢!因此,当使用贪婪的(.+)s修饰符时,它会尽可能匹配直到最后一个与模式匹配的字符吗? - Tony
是的,没错。在找到匹配项之前,它会一直匹配所有内容。 - Wiktor Stribiżew
1个回答

1
如我在评论中提到的那样,贪婪模式需要非常小心谨慎地使用,只有当你需要获得包括子块的最大子字符串时才使用它们。在其他情况下,当你需要单独的子字符串时,请使用懒惰匹配。

这是一个固定的正则表达式

(?>{wrapper btnlabel=(?>"|&quot;)(.+?)(?>"|&quot;)})(.*?){\/wrapper}

示例代码:

$re = "/(?>{wrapper btnlabel=(?>\"|&quot;)(.+?)(?>\"|&quot;)})(.*?){\\/wrapper}/s"; 
$str = "<p>{wrapper btnlabel=&quot;AAA&quot;}zzzzzzzzzzzzzzzzzzzz{/wrapper}</p>\n\n<p>{wrapper btnlabel=&quot;Text&quot;}</p>\n\n<table>\n<tbody>\n<tr>\n<th scope=\"row\">123</th>\n<td>1 123</td>\n<td>12 123</td>\n<td>3 123</td>\n<td>1 123 123</td>\n</tr>\n<tr>\n<th scope=\"row\">123</th>\n<td>2700 123.</td>\n<td>1800 123.</td>\n<td>1000 123.</td>\n<td>300 123.</td>\n</tr>\n</tbody>\n</table>\n\n<p>{/wrapper}</p>"; 
preg_match_all($re, $str, $matches);

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接