我几乎没有替换申请我的$ subject但我不想允许旧替换#(1 .. i-1)的输出与当前替换#i匹配。
$subject1 = preg_replace($pat0, $rep0, $subject0);
$subject2 = preg_replace($pat1, $rep1, $subject1);
$subject3 = preg_replace($pat2, $rep2, $subject2);
我尝试将一个 preg_replace 与数组一起用于模式和替换,希望它能够立即实现;但结果只是连续调用简单的 preg_replace (当然还有一些优化)
在我读到 preg_replace_callback 之后,我想这不是一个解决方案。
任何帮助?
答案 0 :(得分:3)
使用回调,您可以通过使用捕获组来检测匹配的模式,例如(?:(patternt1)|(pattern2)|(etc)
,只会定义匹配模式捕获组。
唯一的问题是你的当前捕获组会被转移。修复(读取变通方法)您可以使用命名组。 (分支重置(?|(foo)|(bar))
将起作用(如果您的版本支持),但是您必须使用其他方式检测哪个模式匹配。)
示例强>
function replace_callback($matches){
if(isset($matches["m1"])){
return "foo";
}
if(isset($matches["m2"])){
return "bar";
}
if(isset($matches["m3"])){
return "baz";
}
return "something is wrong ;)";
}
$re = "/(?|(?:regex1)(?<m1>)|(?:reg(\\s*)ex|2)(?<m2>)|(?:(back refs) work as intended \\1)(?<m3>))/";
$rep_string = preg_replace_callback($re, "replace_callback", $string);
未经过测试(此处没有PHP),但这样的事情可以发挥作用。
答案 1 :(得分:1)
在我看来,preg_replace_callback
是最直接的解决方案。您只需使用|
运算符指定备用模式,并在回调内编码if
或switch
。对我来说似乎是正确的方式。你为什么丢弃它?
另一种解决方案是临时替换特殊字符串。说:
// first pass
$subject = preg_replace($pat0, 'XXX_MYPATTERN0_ZZZ', $subject);
$subject = preg_replace($pat1, 'XXX_MYPATTERN1_ZZZ', $subject);
$subject = preg_replace($pat2, 'XXX_MYPATTERN2_ZZZ', $subject);
// second pass
$subject = preg_replace("XXX_MYPATTERN0_ZZZ",$rep0 , $subject);
$subject = preg_replace("XXX_MYPATTERN1_ZZZ",$rep1 , $subject);
$subject = preg_replace("XXX_MYPATTERN2_ZZZ",$rep2 , $subject);
这非常难看,不能很好地适应动态替换,并且它不是万无一失的,但对于某些“运行一次”的脚本,它可能是可以接受的。