3

$ subjectに適用する置換はほとんどありませんが、古い置換#(1 .. i-1)からの出力を現在の置換#iと一致させたくありません。

$subject1 = preg_replace($pat0, $rep0, $subject0);
$subject2 = preg_replace($pat1, $rep1, $subject1);
$subject3 = preg_replace($pat2, $rep2, $subject2);

パターンと置換の配列で1つの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が最も直接的な解決策のように思えます。|演算子を使用して代替パターンを指定し、コールバック内で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);

これは非常に醜く、動的置換にうまく適応せず、絶対確実ではありませんが、一部の「1回実行」スクリプトでは許容できる場合があります。

于 2011-07-03T20:40:16.280 に答える