程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> PHP正則替換函數preg_replace和preg_replace_callback使用總結

PHP正則替換函數preg_replace和preg_replace_callback使用總結

編輯:PHP綜合

在編寫PHP模板引擎工具類時,以前常用的一個正則替換函數為 preg_replace(),加上正則修飾符 /e,就能夠執行強大的回調函數,實現模板引擎編譯(其實就是字符串替換)。

詳情介紹參考博文:PHP函數preg_replace() 正則替換所有符合條件的字符串

應用舉例如下:
復制代碼 代碼如下:
<?php
/**
 * 模板解析類
 */
class Template {

 public function compile($template) {

  // if邏輯
  $template = preg_replace("/\<\!\-\-\{if\s+(.+?)\}\-\-\>/e", "\$this->ifTag('\\1')", $template);

  return $template;
 }

 /**
  * if 標簽
  */
 protected function ifTag($str) {

  //$str = stripslashes($str); // 去反轉義

  return '<?php if (' . $str . ') { ?>';
 }
}

$template = 'xxx<!--{if $user[\'userName\']}-->yyy<!--{if $user["password"]}-->zzz';

$tplComplier = new Template();

$template = $tplComplier->compile($template);

echo $template;

?>

輸出結果為:
復制代碼 代碼如下:
xxx<?php if ($user['userName']) { ?>yyy<?php if ($user[\"password\"]) { ?>zzz

仔細觀察,發現 $user["password"] 中的雙引號被轉義了,這不是我們想要的結果。

為了能夠正常輸出,還必須反轉義一下,但是,如果字符串中本身含有反轉義雙引號的話,我們此時反轉義,原本的反轉義就變成了非反轉義了,這個結果又不是我們想要的,所以說這個函數在這方面用的不爽!

後來,發現一個更專業級的 正則替換回調函數 preg_replace_callback()。

復制代碼 代碼如下:
mixed preg_replace_callback ( mixed pattern, callback callback, mixed subject [, int limit] )
本函數的行為幾乎和 preg_replace() 一樣,除了不是提供一個 replacement 參數,而是指定一個 callback 函數。該函數將以目標字符串中的匹配數組作為輸入參數,並返回用於替換的字符串。

回調函數 callback:

一個回調函數,在每次需要替換時調用,調用時函數得到的參數是從subject 中匹配到的結果。回調函數返回真正參與替換的字符串。這是該回調函數的簽名:
復制代碼 代碼如下:
string handler ( array $matches )

像上面所看到的,回調函數通常只有一個參數,且是數組類型。

羅列一些有關preg_replace_callback()函數的實例:

Example #1 preg_replace_callback() 和 匿名函數

復制代碼 代碼如下:
<?php
/* 一個unix樣式的命令行過濾器,用於將段落開始部分的大寫字母轉換為小寫。 */
$fp = fopen("php://stdin", "r") or die("can't read stdin");
while (!feof($fp)) {
    $line = fgets($fp);
    $line = preg_replace_callback(
        '|<p>\s*\w|',
        function ($matches) {
            return strtolower($matches[0]);
        },
        $line
    );
    echo $line;
}
fclose($fp);
?>

如果回調函數是個匿名函數,在PHP5.3中,通過關鍵字use,支持給匿名函數傳多個參數,如下所示:

復制代碼 代碼如下:
<?php
$string = "Some numbers: one: 1; two: 2; three: 3 end";
$ten = 10;
$newstring = preg_replace_callback(
    '/(\\d+)/',
    function($match) use ($ten) { return (($match[0] + $ten)); },
    $string
    );
echo $newstring;
#prints "Some numbers: one: 11; two: 12; three: 13 end";
?>

Example #2 preg_replace_callback() 和 一般函數

復制代碼 代碼如下:
<?php
// 將文本中的年份增加一年.
$text = "April fools day is 04/01/2002\n";
$text.= "Last christmas was 12/24/2001\n";
// 回調函數
function next_year($matches) {
  // 通常: $matches[0]是完成的匹配
  // $matches[1]是第一個捕獲子組的匹配
  // 以此類推
  return $matches[1].($matches[2]+1);
}
echo preg_replace_callback(
            "|(\d{2}/\d{2}/)(\d{4})|",
            "next_year",
            $text);

?>

Example #3 preg_replace_callback() 和 類方法

如何在類的內部調用非靜態函數?你可以按如下操作:
對於 PHP 5.2,第二個參數 像這樣 array($this, 'replace') :
復制代碼 代碼如下:
<?php
class test_preg_callback{

  private function process($text){
    $reg = "/\{([0-9a-zA-Z\- ]+)\:([0-9a-zA-Z\- ]+):?\}/";
    return preg_replace_callback($reg, array($this, 'replace'), $text);
  }
 
  private function replace($matches){
    if (method_exists($this, $matches[1])){
      return @$this->$matches[1]($matches[2]);    
    }
  } 
}
?>

對於 PHP5.3,第二個參數像這樣 "self::replace" :
注意,也可以是 array($this, 'replace')。
復制代碼 代碼如下:
<?php
class test_preg_callback{

  private function process($text){
    $reg = "/\{([0-9a-zA-Z\- ]+)\:([0-9a-zA-Z\- ]+):?\}/";
    return preg_replace_callback($reg, "self::replace", $text);
  }
 
  private function replace($matches){
    if (method_exists($this, $matches[1])){
      return @$this->$matches[1]($matches[2]);    
    }
  } 
}
?>


根據上面所學到的知識點,把模板引擎類改造如下:

復制代碼 代碼如下:
<?php
/**
 * 模板解析類
 */
class Template {

 public function compile($template) {

  // if邏輯
  $template = preg_replace_callback("/\<\!\-\-\{if\s+(.+?)\}\-\-\>/", array($this, 'ifTag'), $template);

  return $template;
 }

 /**
  * if 標簽
  */
 protected function ifTag($matches) {
  return '<?php if (' . $matches[1] . ') { ?>';
 }
}

$template = 'xxx<!--{if $user[\'userName\']}-->yyy<!--{if $user["password"]}-->zzz';

$tplComplier = new Template();

$template = $tplComplier->compile($template);

echo $template;

?>

輸出結果為:
復制代碼 代碼如下:
xxx<?php if ($user['userName']) { ?>yyy<?php if ($user["password"]) { ?>zzz

正是我們想要的結果,雙引號沒有被反轉義!

PS:關於正則,本站還提供了2款非常簡便實用的正則表達式工具供大家使用:

JavaScript正則表達式在線測試工具:
http://tools.jb51.net/regex/javascript

正則表達式在線生成工具:
http://tools.jb51.net/regex/create_reg

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved