首页 > 编程技术 > php

正则替换回调函数 preg_replace_callback()—Unicode 转中文方法

发布时间:2017-1-22 11:08

本次一聚教程网为大家带来了正则替换回调函数preg_replace_callback()—Unicode转中文方法,想要学习这个技能的同学快来看看吧。

在之前编写PHP工具类时,常用一个正则替换函数为 preg_replace(),加上正则修饰符 /e,就能够执行强大的回调函数,实现模板引擎编译(也就是字符串替换)。此方法经过多个项目的实验发现存在不足,例如:双引号被转义的问题。

经过文档的查询和学习发现可以使用另外一个正则替换回调函数 preg_replace_callback(),在此做个记录,参考方法如下:

 代码如下复制代码
$data = curl_menu(ACCESS_TOKEN, preg_replace("#\\\u([0-9a-f]+)#ie", "iconv('UCS-2', 'UTF-8', pack('H4', '\\1'))", $data));

替换如下:

 代码如下复制代码
function decodeUnicode($str){
return preg_replace_callback('/\\\\u([0-9a-f]{4})/i', create_function('$matches', 'return iconv("UCS-2BE","UTF-8",pack("H*", $matches[1]));'), $str);
}
小编分享的这篇文章介绍了php如何从一个数组中随机的取出若干个不同的数的教程,有需要的同学可以参考一下

第一种算法,CSDN上别人的想法 

 
 代码如下 复制代码
$num= 0;
 $array=array(1, 2, 3, 4, 5, 6, 7, 8, 9);
 $arr=array();
 $g= 5;
 $tag= true;
 while($tag) {
   $count=count($array);
   $t= rand(0, 1);
   if($t== 1) {
     $arr[] =$array[$num];
     unset($array[$num]);
   }
   $num++;
   if(count($arr) ==$g) {
     $tag= false;
   }
   if($num==$count) {
     $num= 0; //循环
   }
 }
var_dump($arr);
 

 第二种算法,自己想的。

可以在每次取出数据之后将该数据和最后没有获取的数据替换,然后再去没有取得的数据中随机获取值

 
 代码如下 复制代码
functionswap(&$a, &$b)
{
  $temp=$b;
  $b=$a;
  $a=$temp;
}
  
  
$result=array();
$src=array();
for($i= 0 ;$i< 40 ;$i++)
{
  $src[] =$i+ 1;
}
$arr_len=count($src);
$count= 20;
$index= 0;
while($index<$count)
{
  $random= rand(0,$arr_len-$index- 1);
  $result[] =$src[$random];
  swap($src[$random] ,$src[$arr_len-$index- 1]);
  $index+= 1;
}
  
  
print_r(json_encode($result));
print_r(json_encode($src));
 
这篇文章介绍了php中常用的正则函数,对初学者有很好的作用,感兴趣的同学可以参考一下

本文实例总结了php常用正则函数。分享给大家供大家参考,具体如下:

1. mixed preg_replace(mixed pattern, mixed  replacement, mixed  subject, [, int limit])

函数功能:用于正则表达式的搜索和替换。

pattern:正则表达式。
replacement:替换的内容。
subject:需要匹配替换的对象。
limit:可选,指定替换的个数,如果省略 limit 或者其值为 -1,则所有的匹配项都会被替换。

补充说明

① replacement 可以包含 \\n 形式或 $n 形式的逆向引用,首选使用后者。每个此种引用将被替换为与第 n 个被捕获的括号内的子模式所匹配的文本。n 可以从 0 到 99,其中 \\0 或 $0 指的是被整个模式所匹配的文本。对左圆括号从左到右计数(从 1 开始)以取得子模式的数目。

② 对替换模式在一个逆向引用后面紧接着一个数字时(如 \\11),不能使用 \\ 符号来表示逆向引用。因为这样将会使 preg_replace() 搞不清楚是想要一个 \\1 的逆向引用后面跟着一个数字 1 还是一个 \\11 的逆向引用。解决方法是使用 \${1}1。这会形成一个隔离的 $1 逆向引用,而使另一个 1 只是单纯的文字。

③ 上述参数除 limit 外都可以是一个数组。如果 pattern 和 replacement 都是数组,将以其键名在数组中出现的顺序来进行处理,这不一定和索引的数字顺序相同。如果使用索引来标识哪个 pattern 将被哪个 replacement 来替换,应该在调用 preg_replace() 之前用 ksort() 函数对数组进行排序。

例子 1 :

<?php
$str="The quick brown fox jumped over the lazy dog.";
$str= preg_replace('/\s/','-',$str);
echo$str;
?>

输出结果为:

The-quick-brown-fox-jumped-over-the-lazy-dog.

例子 2 ,使用数组:

<?php
$str="The quick brown fox jumped over the lazy dog.";
$patterns[0] ="/quick/";
$patterns[1] ="/brown/";
$patterns[2] ="/fox/";
$replacements[2] ="bear";
$replacements[1] ="black";
$replacements[0] ="slow";
printpreg_replace($patterns,$replacements,$str);
/*输出:
The bear black slow jumped over the lazy dog.
*/
ksort($replacements);
printpreg_replace($patterns,$replacements,$str);
/*输出:
The slow black bear jumped over the lazy dog.
*/
?>

例子 3 ,使用逆向引用:

<?php
$str='<a href="http://www.baidu.com/">baidu</a>其他字符<a href="http://www.sohu.com/">sohu</a>';
$pattern="/<a\s([\s\S]*?)>([\s\S]*?)<\/a>/i";
printpreg_replace($pattern,'\\2',$str);
?>

输出结果为:

baidu其他字符sohu

该例子演示了将文本中所有的 <a></a> 标签去掉。

2. int preg_match(string $pattern,  string  $subject [,array  &$matches [,  int $flags=0 [ ,int $offset=0]]])

函数功能:搜索subject与pattern给定的正则表达式的一个匹配。

pattern:要搜索的模式,字符串类型。
subject:输入字符串。
matches:如果提供了参数matches,它将被填充为搜索结果,$matches[0]将包含完整模式匹配到文本,$matches[1]将包含第一捕获子组匹配到的文本。
flags:可以设置为PREG_OFFSET_CAPTURE,如果传递了这个标记,对于每一个出现的匹配返回时会附加字符串偏移量(相对于目标字符串的)。
注意:这会改变填充到matches数组,使其每个元素成为一个由第0个元素是匹配到的字符串,第1个元素是该匹配字符串在目标字符串subject中的偏移量。
offset:通常,搜索从目标字符串的开始,可选参数offset用于指定从目标字符串的某个未知开始搜索(单位是字节)。

3. int preg_match_all(string $pattern, string $subject [, array &$matches [, int $flags=PREG_PATTERN_ORDER [, int $offset=0]]])

函数功能:搜索subject中所有匹配pattern给定正则表达式的匹配结果并且将它们以flag指定顺序输出到matches中。

在第一个匹配找到后,子序列继续从最后一次匹配位置搜索。

pattern:要搜索的模式,字符串形式。
subject:输入字符串。
matches:多维数组,作为输出参数输出后所有匹配结果,数组排序通过flags指定。
flags:可以结合下面标记使用(注意不能同时使用PREG_PATTERN_ORDER和PREG_SET_ORDER):

PREG_PATTERN_ORDER

结果排序为$matches[0]保存完整模式的所有匹配,$matches[1] 保存第一个子组的所有匹配, 以此类推.

<?php
preg_match_all("|<[^>]+>(.*)</[^>]+>|U",
  "<b>example: </b><div align=left>this is a test</div>",
  $out, PREG_PATTERN_ORDER);
echo$out[0][0] .", ".$out[0][1] ."\n";
echo$out[1][0] .", ".$out[1][1] ."\n";
?>

以上例程会输出:

<b>example: </b>, <div align=left>this is a test</div>
example: , this is a test

因此, $out[0]是包含匹配完整模式的字符串的数组,$out[1]是包含闭合标签内的字符串的数组.

PREG_SET_ORDER

结果排序为$matches[0]包含第一次匹配得到的所有匹配(包含子组),$matches[1]是包含第二次匹配到的所有匹配(包含子组)的数组, 以此类推.

<?php
preg_match_all("|<[^>]+>(.*)</[^>]+>|U",
  "<b>example: </b><div align=\"left\">this is a test</div>",
  $out, PREG_SET_ORDER);
echo$out[0][0] .", ".$out[0][1] ."\n";
echo$out[1][0] .", ".$out[1][1] ."\n";
?>

以上例程会输出:

<b>example: </b>, example:
<div align="left">this is a test</div>, this is a test

PREG_OFFSET_CAPTURE

如果这个标记被传递, 每个发现的匹配返回时会增加它相对目标字符串的偏移量. 注意这会改变matches中的每一个匹配结果字符串元素, 使其 成为一个第0个元素为匹配结果字符串, 第1个元素为 匹配结果字符串在subject中的偏移量.

如果没有给定排序标记, 假定设置为PREG_PATTERN_ORDER.

offset:通常,查找时从目标字符串的开始位置开始,可选参数offset用于从目标字符串中指定位置开始搜索(单位是字节)。

这篇文章详解了PHP中EventDispatcher事件分发组件,插件可以添加一些方法,或者在某些方法执行之前或者之后做些事情,而不干扰其他插件。有需要的同学可以参考一下

引言

考虑这样一个问题,现在你想给为你的项目提供一个插件系统,插件可以添加一些方法,或者在某些方法执行之前或者之后做些事情,而不干扰其他插件。要实现这个系统,简单的单继承不是个好办法,即使多继承在PHP中是可能的,他也有与生俱来的缺点(多继承不太了解,感觉挺操蛋的)。

Symfony EventDispatcher以一个简单有效的方式实现了中介者模式,事件分发器就是那个中介,让系统和插件不会耦合在一起,这让上面的插件系统成为可能,而且他会让你的项目可扩展性更好。

上面的话,翻译自Symfony官方文档片段

系统剖析

事件存储

上面这张图是分析Symfony EventDispatcher组件源码得出来的,可以看到事件在系统中是如何存储的

这里面将事件存储了两遍,用来加入优先级priority的概念,存如的时候放入上图中上面的结构中,取出时候从上图中下面的结构中拿出来,相同的事件名称可以有不同的优先级,优先级越高的事件优先触发,优先级相同的时候,先插入的事件优先触发。

排序事件(上图中下面的结构)在插入事件的时候不会构建,而是当取出事件的时候会生成排好序的事件,当相同的事件名中插入新的事件或删除某个事件的时候,会删除对应的排好序的事件名,后面用到的时候重新构建

执行事件的时候,会获取对应事件名排好序的linster列表,按照顺序依次执行。

事件执行

如上图所示,当触发某个时间的时候,该事件名下面如果监听了多个触发动作,他们会按照优先级、注册顺序依次触发,触发动作一般是一个可执行的“实例”(不管是类还是函数,必须可以通过call_user_func_array调用),可以传入三个参数,第一个参数(必须)是一个Event实例,第二个是触发的事件名,第三个是事件分发器实例。第一个参数会控制事件是否在该事件名下的所有触发动作之间继续传递,比如上面的linstener_2里面将Event.propagationStopped设置为true,执行完linstener_2后,事件就会停止传播,linstener_2后面的动作不会触发。

除此之外,Event实例中还可以保存其他必要的信息,以便linstener触发执行的时候,获取额外的信息。

事件订阅者

事件订阅者(Event subscriber),告诉dispathcer实例,他要订阅的所有事件,不用一个个通过dispathcer实例去注册。事件订阅者是一个PHP类,他可以告诉dispathcer他要订阅的具体的事件。

好处:

订阅者内部关注的事件是一个整体,要么全部关注要么全部不关注

实例

普通栗子

include"vendor/autoload.php";
useSymfony\Component\EventDispatcher\EventDispatcher;
useSymfony\Component\EventDispatcher\Event;
classUserEventextendsEvent
{
 publicfunctionname()
 {
 return"Cartman";
 }
 
 publicfunctionage()
 {
 return"24";
 }
}
$dispatcher=newEventDispatcher();
$dispatcher->addListener("user.name",function($event,$eventName,$dispatcher){
 echo"My name is Cartman\n";
});
$dispatcher->addListener("user.name",function($event,$eventName,$dispatcher){
 echo"My name is {$event->name()} from Event instance\n";
}, 10);
$dispatcher->addListener("user.age",function($event,$eventName,$dispatcher){
 echo"My age is 24\n";
}, 10);
$dispatcher->addListener("user.age",function($event,$eventName,$dispatcher){
 echo"My age is {$event->age()} from Event instance\n";
}, -10);
$dispatcher->dispatch("user.name",newUserEvent());
$dispatcher->dispatch("user.age",newUserEvent());

上面的例子输出

My name is Cartman from Event instance
My name is Cartman
My age is 24
My age is 24 from Event instance

事件订阅者栗子

通过Subscriber注册事件

include"vendor/autoload.php";
useSymfony\Component\EventDispatcher\EventDispatcher;
useSymfony\Component\EventDispatcher\Event;
useSymfony\Component\EventDispatcher\EventSubscriberInterface;
classBookEventextendsEvent
{
 public$name= self::class;
}
classBookSubscriberimplementsEventSubscriberInterface
{
 publicstaticfunctiongetSubscribedEvents()
 {
 return[
  "chinese.name"=>"chineseNameShow",
  "english.name"=> [
  ["englishNameShow", -10],
  ["englishNameAFter", 10],
  ],
  "math.name"=> ["mathNameShow", 100]
 ];
 }
 publicfunctionchineseNameShow(Event$event)
 {
 echo"我是汉语书籍\n";
 }
 publicfunctionenglishNameShow(Event$event)
 {
 echo"我是英文书籍\n";
 }
 publicfunctionenglishNameAFter(Event$event)
 {
 echo"我是展示之后的英文书籍[来自于Event实例{$event->name}]\n";
 }
 publicfunctionmathNameShow(Event$event)
 {
 echo"我是展示的数学书籍\n";
 }
}
$dispatcher=newEventDispatcher();
$subscriber=newBookSubscriber();
$dispatcher->addSubscriber($subscriber);
$dispatcher->dispatch("english.name",newBookEvent());
$dispatcher->dispatch("chinese.name");
$dispatcher->removeSubscriber($subscriber);
$dispatcher->dispatch("math.name");

输出为内容:

我是展示之后的英文书籍[来自于Event实例BookEvent]
我是英文书籍
我是汉语书籍
标签:[!--infotagslink--]

您可能感兴趣的文章: