VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > PHP >
  • PHP正则替换函数preg_replace和preg_replace_callback使用总结

这篇文章主要介绍了PHP正则替换函数preg_replace和preg_replace_callback使用总结,本文是在写一个模板引擎遇到一个特殊需求时总结而来,需要的朋友可以参考下

在编写PHP模板引擎工具类时,以前常用的一个正则替换函数为 preg_replace(),加上正则修饰符 /e,就能够执行强大的回调函数,实现模板引擎编译(其实就是字符串替换)。

应用举例如下:

  1. <?php 
  2. /** 
  3.  * 模板解析类 
  4.  */ 
  5. class Template { 
  6.  public function compile($template) { 
  7.  
  8.   // if逻辑 
  9.   $template = preg_replace("/\<\!\-\-\{if\s+(.+?)\}\-\-\>/e""\$this->ifTag('\\1')"$template); 
  10.  
  11.   return $template
  12.  } 
  13.  
  14.  /** 
  15.   * if 标签 
  16.   */ 
  17.  protected function ifTag($str) { 
  18.  
  19.   //$str = stripslashes($str); // 去反转义 
  20.  
  21.   return '<?php if (' . $str . ') { ?>'
  22.  } 
  23.  
  24. $template = 'xxx<!--{if $user[\'userName\']}-->yyy<!--{if $user["password"]}-->zzz'
  25.  
  26. $tplComplier = new Template(); 
  27.  
  28. $template = $tplComplier->compile($template); 
  29.  
  30. echo $template
  31.  
  32. ?> 

输出结果为:

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() 和 匿名函数

代码如下:

  1. <?php 
  2. /* 一个unix样式的命令行过滤器,用于将段落开始部分的大写字母转换为小写。 */ 
  3. $fp = fopen("php://stdin""r"or die("can't read stdin"); 
  4. while (!feof($fp)) { 
  5.     $line = fgets($fp); 
  6.     $line = preg_replace_callback( 
  7.         '|<p>\s*\w|'
  8.         function ($matches) { 
  9.             return strtolower($matches[0]); 
  10.         }, 
  11.         $line 
  12.     ); 
  13.     echo $line
  14. fclose($fp); 
  15. ?> 

如果回调函数是个匿名函数,在PHP5.3中,通过关键字use,支持给匿名函数传多个参数,如下所示:

  1. <?php 
  2. $string = "Some numbers: one: 1; two: 2; three: 3 end"
  3. $ten = 10; 
  4. $newstring = preg_replace_callback( 
  5.     '/(\\d+)/'
  6.     function($matchuse ($ten) { return (($match[0] + $ten)); }, 
  7.     $string 
  8.     ); //www.phpfensi.com 
  9. echo $newstring
  10. #prints "Some numbers: one: 11; two: 12; three: 13 end"
  11. ?> 

Example #2 preg_replace_callback() 和 一般函数

代码如下:

  1. <?php 
  2. // 将文本中的年份增加一年. 
  3. $text = "April fools day is 04/01/2002\n"
  4. $text.= "Last christmas was 12/24/2001\n"
  5. // 回调函数 
  6. function next_year($matches) { 
  7.   // 通常: $matches[0]是完成的匹配 
  8.   // $matches[1]是第一个捕获子组的匹配 
  9.   // 以此类推 
  10.   return $matches[1].($matches[2]+1); 
  11. echo preg_replace_callback( 
  12.             "|(\d{2}/\d{2}/)(\d{4})|"
  13.             "next_year"
  14.             $text); 
  15. ?> 

Example #3 preg_replace_callback() 和 类方法

如何在类的内部调用非静态函数?你可以按如下操作:

对于 PHP 5.2,第二个参数 像这样 array($this, 'replace') :

  1. <?php 
  2. class test_preg_callback{ 
  3.   private function process($text){ 
  4.     $reg = "/\{([0-9a-zA-Z\- ]+)\:([0-9a-zA-Z\- ]+):?\}/"
  5.     return preg_replace_callback($regarray($this'replace'), $text); 
  6.   } 
  7.  
  8.   private function replace($matches){ 
  9.     if (method_exists($this$matches[1])){ 
  10.       return @$this->$matches[1]($matches[2]);     
  11.     } 
  12.   }  
  13. ?> 

对于 PHP5.3,第二个参数像这样 "self::replace" :

注意,也可以是 array($this, 'replace')。

代码如下:

  1. <?php 
  2. class test_preg_callback{ 
  3.   private function process($text){ 
  4.     $reg = "/\{([0-9a-zA-Z\- ]+)\:([0-9a-zA-Z\- ]+):?\}/"
  5.     return preg_replace_callback($reg"self::replace"$text); 
  6.   } 
  7.  
  8.   private function replace($matches){ 
  9.     if (method_exists($this$matches[1])){ 
  10.       return @$this->$matches[1]($matches[2]);     
  11.     } 
  12.   }  
  13. ?> 

根据上面所学到的知识点,把模板引擎类改造如下:

  1. <?php 
  2. /** 
  3.  * 模板解析类 
  4.  */ 
  5. class Template { 
  6.  public function compile($template) { 
  7.  
  8.   // if逻辑 
  9.   $template = preg_replace_callback("/\<\!\-\-\{if\s+(.+?)\}\-\-\>/"array($this'ifTag'), $template); 
  10.  
  11.   return $template
  12.  } 
  13.  
  14.  /** 
  15.   * if 标签 
  16.   */ 
  17.  protected function ifTag($matches) { 
  18.   return '<?php if (' . $matches[1] . ') { ?>'
  19.  } 
  20. //www.phpfensi.com 
  21. $template = 'xxx<!--{if $user[\'userName\']}-->yyy<!--{if $user["password"]}-->zzz'
  22.  
  23. $tplComplier = new Template(); 
  24.  
  25. $template = $tplComplier->compile($template); 
  26.  
  27. echo $template
  28.  
  29. ?> 

输出结果为:

xxx<?php if ($user['userName']) { ?>yyy<?php if ($user["password"]) { ?>zzz

正是我们想要的结果,双引号没有被反转义!

出处:http://www.phpfensi.com/php/20210412/14244.html


相关教程