3

我的 $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之后,我想这不是一个解决方案。

有什么帮助吗?

4

2 回答 2

5

使用回调,您可以使用捕获组检测匹配的模式,例如(?:(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),但这样的东西可以工作。

于 2011-07-03T21:01:26.673 回答
1

在我看来,这preg_replace_callback是最直接的解决方案。您只需使用|运算符指定备用模式,并在回调中编写ifor 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);

这很丑陋,不能很好地适应动态替换,而且它也不是万无一失的,但对于一些“运行一次”的脚本来说它可能是可以接受的。

于 2011-07-03T20:40:16.280 回答