Прежде всего, я хочу извиниться, если есть существующая тема, я много искал, но не смог найти решение.
Итак, у меня есть функция preg_replace_callback, которая заменяет определенные теги в строке функцией.
Пример:
$object = preg_replace_callback('~{FUNC\s+(.+?)}(.+?){/FUNC}~is', function($matches) use ($replace)
{
list($condition, $function, $content) = $matches;
return $function($content);
}, $object);
Но когда я использую вложенный тег внутри другого тега, он терпит неудачу
Пример:
{FUNC name_of_func}
text
text
text
{FUNC name_of_func2}text 2{/FUNC}
text
text
{/FUNC}
Я знаю, что он находит первый закрывающий тег, и в этом проблема, но я плохо работаю с регулярным выражением, как это исправить, чтобы я мог использовать несколько вложенных тегов или вложенных вложенных тегов, если это возможно?






Здесь мы можем захотеть найти открывающие и закрывающие теги, которые мы хотим заменить, собрать их с помощью preg_match_all, затем мы заменим их один за другим на preg_regplace на основе наших шаблонов, например:
$re = '/{FUNC\s+(.+?)}|{/FUNC}/mi';
$str = '{FUNC name_of_func}
text
text
text
{FUNC name_of_func2}text 2{/FUNC}
text
text
{/FUNC}';
preg_match_all($re, $str, $matches, PREG_SET_ORDER, 0);
foreach ($matches as $key => $match) {
if ($match[1] != null) {
$str = preg_replace('/({FUNC\s+)(.+?)(})/is', '$1Another new name that we wish$3', $str);
}
if (preg_match('/{[a-z]/is', $match[0])) {
$str = preg_replace($match[0], '{New_FUNC}', $str);
} elseif (preg_match('/{//s', $match[0])) {
$str = preg_replace($match[0], '/New_CLOS_FUNC', $str);
} else {
continue;
}
}
var_dump($str);
{FUNC Another new name that we wish}
text
text
text
{FUNC Another new name that we wish}text 2{/New_CLOS_FUNC}
text
text
{/New_CLOS_FUNC}
jex.im визуализирует регулярные выражения:
Чтобы выполнить пользовательские замены в конечном итоге вложенных структур с помощью preg_replace_callback, простой способ состоит в том, чтобы сначала заменить самые внутренние части в цикле while, пока не будет ничего заменять. Для этого ваш шаблон должен запрещать вложенные части.
Во-вторых, вместо того, чтобы без необходимости копировать массивы совпадений с помощью list(), лучше использовать именованные захваты:
$replace = [
'func1' => function ($var) { /*...*/ },
'func2' => function ($var) { /*...*/ },
//...
];
$pattern = '~{FUNC \s+ (?<funcName> \w+ )}
(?<content> [^{]*+ (?: { (?!/?FUNC\b) [^{]* )*+ )
{/FUNC}~xi';
do {
$object = preg_replace_callback($pattern, function($m) use ($replace) {
return $replace[$m['funcName']]($m['content']);
}, $object, -1, $count);
} while ($count);
Это из-за вашего ленивого квантификатора, который останавливается на первом встреченном
{/FUNC}. При наличии вложенных конструкций используйте либо рекурсивный подход, либо синтаксический анализатор.