首页 > 作文

PHP标准库(PHP SPL)详解

更新时间:2023-04-06 14:46:11 阅读: 评论:0

什么是spl?

spl,php 标准库(standard php library) ,此从 php 5.0 起内置的组件和接口,并且从 php5.3 已逐渐的成熟。spl 其实在所有的 php5 开发环境中被内置,同时无需任何设置。

似乎众多的 php 开发人员基本没有使用它,甚至闻所未闻。究其原因,可以追述到它那阳春白雪般的说明文档,使你忽略了「它的存在」。spl 这块宝石犹如铁达尼的「海洋之心」般,被沉入海底。而现在它应该被我们捞起,并将它穿戴在应有的位置 ,而这也是这篇文章所要表述的观点。

那么,spl 提供了什么?

spl 对 php 引擎进行了扩展,例如 arrayaccess、countable 和 ekableiterator 等接口,它们用于以数组形式操作对象。同时,你还可以使用 recursiveiterator、arrayobejcts 等其他迭代器进行数据的迭代操作。

它还内置几个的对象例如 exceptions、splobrver、spltorage 以及 splautoloadregister、splclass、iterat英语语法结构orapply 等的帮助函数(helper functions),用于重载对应的功能。

这些工具聚合在一起就好比是把多功能的瑞士军刀,善用它们可以从质上提升 php 的代码效率。那么,我们如何发挥它的威力?

如何使用spl?

spl提供了一组标准数据结构:

双向链表

spldoublylinkedlist

splstacksplqueue

双链表是一种重要的线性存储结构,对于双链表中的每个节点,不仅仅存储自己的信息,还要保存前驱和后继节点的地址。

php spl中的spldoublylinkedlist类提供了对双链表的操作。

spldoublylinkedlist类摘要如下:

 spldoublylinkedlist implements iterator , arrayaccess , countable {   public __construct ( void )  public void add ( mixed $index , mixed $newval )  //双链表的头部节点  public mixed top ( void )  //双链表的尾部节点  public mixed bottom ( void )  //双联表元素的个数  public int count ( void )  //检测双链表是否为空  public bool impty ( void )  //当前节点索引  public mixed key ( void )  //移到上条记录  public void prev ( void )  //移到下条记录  public void next ( void )  //当前记录  public mixed current ( void )  //将指针指向迭代开始处  public void rewind ( void )  //检查双链表是否还有节点  public bool valid ( void )    //指定index处节点是否存在  public bool offtexists ( mixed $index )  //获取指定index处节点值  public mixed offtget ( mixed $index )  //设置指定index处值  public void offtt ( mixed $index , mixed $newval )  //删除指定index处节点  public void offtunt ( mixed $index )   //从双链表的尾部弹出元素  public mixed pop ( void )  //添加元素到双链表的尾部  public void push ( mixed $value )    //序列化存储  public string rialize ( void )  //反序列化  public void unrialize ( string $rialized )  //设置迭代模式  public void titeratormode ( int $mode )  //获取迭代模式spldoublylinkedlist::it_mode_lifo (stack style) spldoublylinkedlist::it_mode_fifo (queue style)  public int getiteratormode ( void )  //双链表的头部移除元素  public mixed shift ( void )  //双链表的头部添加元素  public void unshift ( mixed $value ) }

使用起来也比较简单

$list = new spldoublylinkedlist(); $list->push('a'); $list->push('b'); $list->push('c'); $list->push('d');  $list->unshift('top'); $list->shift();  $list->rewind();//rewind操作用于把节点指针指向bottom所在的节点 echo 'curren node:'.$list->current()."<br />";//获取当前节点 $list->next();//指针指向下一个节点 echo 'next node:'.$list->current()."<br />"; $list->next(); $list->next(); $list->prev();//指针指向上一个节点 echo 'next node:'.$list->current()."<br />"; if($list->current())   echo 'current node is valid<br />'; el   echo 'current node is invalid&l秘密花园好词好句t;br />'; if($list->valid())//如果当前节点是有效节点,valid返回true   echo "valid list<br />"; el   echo "invalid list <br />"; var_dump(array(   'pop' => $list->pop(),   'count' => $list->count(),   'impty' => $list->impty(),   'bottom' => $list->bottom(),   'top' => $list->top() )); $list->titeratormode(spldoublylinkedlist::it_mode_fifo); var_dump($list->getiteratormode()); for($list->rewind(); $list->valid(); $list->next()) {   echo $list->current().php_eol; } var_dump($a = $list->rialize()); //print_r($list->unrialize($a)); $list->offtt(0,'new one'); $list->offtunt(0); var_dump(array(   'offtexists' => $list-&g孩子容易走神t;offtexists(4),   'offtget' => $list-&g重庆专科院校t;offtget(0), )); var_dump($list); //堆栈,先进后出 $stack = new splstack();//继承自spldoublylinkedlist类 $stack->push("a<br />"); $stack->push("b<br />"); echo $stack->pop(); echo $stack->pop(); echo $stack->offtt(0,'b');//堆栈的offt=0是top所在的位置,offt=1是top位置节点靠近bottom位置的相邻节点,以此类推 $stack->rewind();//双向链表的rewind和堆栈的rewind相反,堆栈的rewind使得当前指针指向top所在的位置,而双向链表调用之后指向bottom所在位置 echo 'current:'.$stack->current().'<br />'; $stack->next();//堆栈的next操作使指针指向靠近bottom位置的下一个节点,而双向链表是靠近top的下一个节点 echo 'current:'.$stack->current().'<br />'; echo '<br /><br />'; //队列,先进先出 $queue = new splqueue();//继承自spldoublylinkedlist类 $queue->enqueue("a<br />");//插入一个节点到队列里面的top位置 $queue->enqueue("b<br />"); $queue->offtt(0,'a');//堆栈的offt=0是top所在的位置,offt=1是top位置节点靠近bottom位置的相邻节点,以此类推 echo $queue->dequeue(); echo $queue->dequeue(); echo "<br /><br />";

重载 autoloader

如果你是位「教科书式的程序员」,那么你保证了解如何使用 __autoload 去代替 includes/requires 操作惰性载入对应的类,对不?

但久之,你会发现你已经陷入了困境,首先是你要保证你的类文件必须在指定的文件路径中,例如在 zend 框架中你必须使用「_」来分割类、方法名称(你如何解决这一问题?)。

另外的一个问题,就是当项目变得越来越复杂, __autoload 内的逻辑也会变得相应的复杂。到最后,甚至你会加入异常判断,以及将所有的载入类的逻辑如数写到其中。

大家都知道「鸡蛋不能放到一个篮子中」,利用 spl 可以分离 __autoload 的载入逻辑。只需要写个你自己的 autoload 函数,然后利用 spl 提供的函数重载它这世界需要你 作文。

例如上述 zend 框架的问题,你可以重载 zend loader 对应的方法,如果它没有找到对应的类,那么就使用你先前定义的函数。

<?phpclass myloader {  public static function doautoload($class) {    // 本模块对应的 autoload 操作  }}spl_autoload_register( array('myloader', 'doautoload') );?>

正如你所见, spl autoload register 还能以数组的形式加入多个载入逻辑。同时,你还可以利用spl autoload unregister 移除已经不再需要的载入逻辑,这功能总会用到的。

迭代器

迭代是常见设计模式之一,普遍应用于一组数据中的统一的遍历操作。可以毫不夸张的说,spl 提供了所有你需要的对应数据类型的迭代器。

有个非常好的案例就是遍历目录。常规的做法就是使用 scandir ,然后跳过「.「 和 「..」,以及其它未满足条件的文件。例如你需要遍历个某个目录抽取其中的图片文件,就需要判断是否是 jpg、gif 结尾。

下面的代码就是使用 spl 的迭代器执行上述递归寻找指定目录中的图片文件的例子:

<?phpclass recursivefilefilteriterator extends filteriterator {  // 满足条件的扩展名  protected $ext = array('jpg','gif');  /**   * 提供 $path 并生成对应的目录迭代器   */  public function __construct($path) {    parent::__construct(new recursiveiteratoriterator(new recursivedirectoryiterator($path)));  }  /**   * 检查文件扩展名是否满足条件   */  public function accept() {    $item = $this->getinneriterator();    if ($item->isfile() &&         in_array(pathinfo($item->getfilename(), pathinfo_extension), $this->ext)) {      return true;    }  }}// 实例化foreach (new recursivefilefilteriterator('/path/to/something') as $item) {  echo $item . php_eol;}?>

你可能会说,这不是花了更多的代码去办同一件事情吗?那么,查看上面的代码,你不是拥有了具有高度重用而且可以测试的代码了吗

本文发布于:2023-04-06 14:46:09,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/zuowen/6d9d835542b403655e85394e5e3cc4d1.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:PHP标准库(PHP SPL)详解.doc

本文 PDF 下载地址:PHP标准库(PHP SPL)详解.pdf

标签:节点   堆栈   链表   数组
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图