У меня есть такая строка в базе данных (фактическая строка содержит 100 слов и 10 переменных)
i am a {$club} fan
я повторяю эту строку, как это
$club = "Barcelona";
echo $data_base[0]['body'];
Мой вывод i am a {$club} fan
,
я хочу i am a Barcelona fan
Как я могу это сделать?
использование strtr
Он переведет части строки.
$club = "Barcelona";
echo strtr($data_base[0]['body'], array('{$club}' => $club));
Для нескольких значений (демонстрация):
$data_base[0]['body'] = 'I am a {$club} fan.'; // Tests
$vars = array(
'{$club}' => 'Barcelona',
'{$tag}' => 'sometext',
'{$anothertag}' => 'someothertext'
);
echo strtr($data_base[0]['body'], $vars);
Выход программы:
I am a Barcelona fan.
/**
* A function to fill the template with variables, returns filled template.
*
* @param string $template A template with variables placeholders {$varaible}.
* @param array $variables A key => value store of variable names and values.
*
* @return string
*/
public function replaceVariablesInTemplate($template, array $variables){
return preg_replace_callback('#{(.*?)}#',
function($match) use ($variables){
$match[1]=trim($match[1],'$');
return $variables[$match[1]];
},
' '.$template.' ');
}
if (preg_match_all('#\$([a-zA-Z0-9]+)#', $q, $matches, PREG_SET_ORDER));
{
foreach ($matches as $m)
{
eval('$q = str_replace(\''.$m[0].'\',$'.$m[1].',$q);');
}
}
Это соответствует всем переменным $ и заменяет их значением.
я не включил {}, но не должно быть слишком сложно, чтобы добавить что-то вроде этого …
if (preg_match_all('#\{\$([a-zA-Z0-9]+)\}#', $q, $matches, PREG_SET_ORDER));
{
foreach ($matches as $m)
{
eval('$q = str_replace(\''.$m[0].'\',$'.$m[1].',$q);');
}
}
Хотя это кажется немного медленнее, чем жесткое кодирование каждой переменной. И это вводит дыру в безопасности с eval. Вот почему мое регулярное выражение так ограничено. Чтобы ограничить сферу действия eval.
Я написал свой собственный тестер регулярных выражений с помощью ajax, чтобы по мере ввода текста я мог видеть, сработает ли мое выражение. У меня есть переменные, которые я хотел бы использовать в своих выражениях, чтобы мне не нужно было повторять один и тот же бит для каждого выражения.
Я бы предложил sprintf()
функция.
вместо хранения i am a {$club} fan
использование i am a %s fan
поэтому ваша команда echo будет выглядеть так:
$club = "Barcelona";
echo sprintf($data_base[0]['body'],$club);
ВЫХОД: я фанат Барселоны
Это дало бы вам свободу использования того же кода с любой другой переменной (и вам даже не нужно запоминать имя переменной).
поэтому этот код также действителен с той же строкой:
$food = "French Fries";
echo sprintf($data_base[0]['body'],$food);
ВЫХОД: я фанат картофеля фри
$language = "PHP";
echo sprintf($data_base[0]['body'],$language);
ВЫВОД: я фанат PHP
То, что вы ищете, это интерполяция вложенных строк. Теорию можно прочитать в этом посте: http://thehighcastle.com/blog/21/wanted-php-core-function-for-dynamically-performing-double-quoted-string-variable-interpolation
Основная проблема заключается в том, что вы на самом деле не знаете все доступные переменные, или их может быть слишком много для перечисления.
Рассмотрим следующий проверенный фрагмент кода. Я украл регулярное выражение у Мохаммада Мохсенипура.
$testA = '123';
$testB = '456';
$testC = '789';
$t = '{$testA} adsf {$testB}adf 32{$testC} fddd{$testA}';
echo 'before: '.$t."\n";
preg_match_all('~\{\$(.*?)\}~si',$t,$matches);
if ( isset($matches[1])) {
$r = compact($matches[1]);
foreach ( $r as $var => $value ) {
$t = str_replace('{$'.$var.'}',$value,$t);
}
}
echo 'after: '.$t."\n";
Ваш код может быть:
$club = 'Barcelona';
$tmp = $data_base[0]['body'];
preg_match_all('~\{\$(.*?)\}~si',$tmp,$matches);
if ( isset($matches[1])) {
$r = compact($matches[1]);
foreach ( $r as $var => $value ) {
$tmp = str_replace('{$'.$var.'}',$value,$tmp);
}
}
echo $tmp;
вот мое решение:
$club = "Barcelona";
$string = 'i am a {$club} fan';
preg_match_all("/\{\\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\}/",$string,$matches);
foreach ($matches[0] as $key => $var_name) {
if (!isset($GLOBALS[$matches[1][$key]])) $GLOBALS[$matches[1][$key]] = 'default value';
$string = str_replace($var_name, $GLOBALS[$matches[1][$key]], $string);
}
Вы можете использовать простой парсер, который заменяет {$ key} значением из карты, если оно существует.
использовать как $text = templateWith('hello $item}',array('item'=>'world'...));
Моя первая версия:
/**
* Template with a string and simple map.
* @param string $template
* @param array $substitutions map of substitutions.
* @return string with substitutions applied.
*/
function templateWith(string $template, array $substitutions) {
$state = 0; // forwarding
$charIn = preg_split('//u', $template, -1, PREG_SPLIT_NO_EMPTY);
$charOut = array();
$count = count($charIn);
$key = array();
$i = 0;
while ($i < $count) {
$char = $charIn[$i];
switch ($char) {
case '{':
if ($state === 0) {
$state = 1;
}
break;
case '}':
if ($state === 2) {
$ks = join('', $key);
if (array_key_exists($ks, $substitutions)) {
$charOut[] = $substitutions[$ks];
}
$key = array();
$state = 0;
}
break;
case '$': if ($state === 1) {
$state = 2;
}
break;
case '\\': if ($state === 0) {
$i++;
$charOut[] = $charIn[$i];
}
continue;
default:
switch ($state) {
default:
case 0: $charOut[] = $char;
break;
case 2: $key[] = $char;
break;
}
}
$i++;
}
return join('', $charOut);
}
Пытаться preg_replace
функция php.
http://php.net/manual/en/function.preg-replace.php
<?php
$club = "barcelona";
echo $string = preg_replace('#\{.*?\}#si', $club, 'i am a {$club} fan');
?>