PHP标准库(PHP SPL)详解


Posted in PHP onMarch 16, 2019

什么是SPL?

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

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

那么,SPL 提供了什么?

SPL 对 PHP 引擎进行了扩展,例如 ArrayAccess、Countable 和 SeekableIterator 等接口,它们用于以数组形式操作对象。同时,你还可以使用 RecursiveIterator、ArrayObejcts 等其他迭代器进行数据的迭代操作。

它还内置几个的对象例如 Exceptions、SplObserver、Spltorage 以及 splautoloadregister、splclasses、iteratorapply 等的帮助函数(helper functions),用于重载对应的功能。

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

如何使用SPL?

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

双向链表

SplDoublyLinkedList

  • SplStack
  • SplQueue

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

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 isEmpty ( 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 offsetExists ( mixed $index )
  //获取指定index处节点值
  public mixed offsetGet ( mixed $index )
  //设置指定index处值
  public void offsetSet ( mixed $index , mixed $newval )
  //删除指定index处节点
  public void offsetUnset ( mixed $index ) 
  //从双链表的尾部弹出元素
  public mixed pop ( void )
  //添加元素到双链表的尾部
  public void push ( mixed $value )  
  //序列化存储
  public string serialize ( void )
  //反序列化
  public void unserialize ( string $serialized )
  //设置迭代模式
  public void setIteratorMode ( 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 />';
 else
   echo 'current node is invalid<br />';
 if($list->valid())//如果当前节点是有效节点,valid返回true
   echo "valid list<br />";
 else
   echo "invalid list <br />";
 var_dump(array(
   'pop' => $list->pop(),
   'count' => $list->count(),
   'isEmpty' => $list->isEmpty(),
   'bottom' => $list->bottom(),
   'top' => $list->top()
 ));
 $list->setIteratorMode(SplDoublyLinkedList::IT_MODE_FIFO);
 var_dump($list->getIteratorMode());
 for($list->rewind(); $list->valid(); $list->next()) {
   echo $list->current().PHP_EOL;
 }
 var_dump($a = $list->serialize());
 //print_r($list->unserialize($a));
 $list->offsetSet(0,'new one');
 $list->offsetUnset(0);
 var_dump(array(
   'offsetExists' => $list->offsetExists(4),
   'offsetGet' => $list->offsetGet(0),
 ));
 var_dump($list);
 //堆栈,先进后出
 $stack = new SplStack();//继承自SplDoublyLinkedList类
 $stack->push("a<br />");
 $stack->push("b<br />");
 echo $stack->pop();
 echo $stack->pop();
 echo $stack->offsetSet(0,'B');//堆栈的offset=0是Top所在的位置,offset=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->offsetSet(0,'A');//堆栈的offset=0是Top所在的位置,offset=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 对应的方法,如果它没有找到对应的类,那么就使用你先前定义的函数。

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

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

迭代器

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

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

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

<?php
class 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;
}
?>

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

下面是 SPL 提供的其他的迭代器:

  • RecursiveIterator
  • RecursiveIteratorIterator
  • OuterIterator
  • IteratorIterator
  • FilterIterator
  • RecursiveFilterIterator
  • ParentIterator
  • SeekableIterator
  • LimitIterator
  • GlobIterator
  • CachingIterator
  • RecursiveCachingIterator
  • NoRewindIterator
  • AppendIterator
  • RecursiveIteratorIterator
  • InfiniteIterator
  • RegexIterator
  • RecursiveRegexIterator
  • EmptyIterator
  • RecursiveTreeIterator
  • ArrayIterator

自 PHP5.3 开始,会内置其他更多的迭代器,我想你都可以尝试下,或许它能改变你编写传统代码的习惯。

SplFixedArray

SPL 还内置了一系列的数组操作工具,例如可以使用 SplFixedArray 实例化一个固定长度的数组。那么为什么要使用它?因为它更快,甚至它关系着你的工资问题 :)

我们知道 PHP 常规的数组包含不同类型的键,例如数字、字符串等,并且长度是可变的。正是因为这些「高级功能」,PHP 以散列(hash)的方式通过键得到对应的值 -- 其实这在特定情况这会造成性能问题。

而 SplFixedArray 因为是使用固定的数字键,所以它并没有使用散列存储方式。不确切的说,甚至你可以认为它就是个 C 数组。这就是为什么 SplFixedArray 会比通常数组要快的原因(仅在 PHP5.3 中)。

那到底有多快呢,下面的组数据可以让你窥其究竟。

PHP标准库(PHP SPL)详解

如果你需要大量的数组操作,那么你可以尝试下,相信它是值得信赖的。

数据结构

同时 SPL 还提供了些数据结构基本类型的实现 。虽然我们可以使用传统的变量类型来描述数据结构,例如用数组来描述堆栈(Strack)-- 然后使用对应的方式 pop 和 push(arraypop()、arraypush()),但你得时刻小心,·因为毕竟它们不是专门用于描述数据结构的 -- 一次误操作就有可能破坏该堆栈。

而 SPL 的 SplStack 对象则严格以堆栈的形式描述数据,并提供对应的方法。同时,这样的代码应该也能理解它在操作堆栈而非某个数组,从而能让你的同伴更好的理解相应的代码,并且它更快。

最后,可能上述那些惨白的例子还不足矣「诱惑你」去使用 SPL。实践出真知,SPL 更多、更强大的功能需要你自己去挖掘。而它正如宝石般的慢慢雕砌,才能散发光辉。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。如果你想了解更多相关内容请查看下面相关链接

PHP 相关文章推荐
特转载一高手总结PHP学习资源和链接.
Dec 05 PHP
Wordpress 相册插件 NextGEN-Gallery 添加目录将中文转为拼音的解决办法
Dec 29 PHP
使用PHP遍历文件夹与子目录的函数代码
Sep 26 PHP
php 字符串中的\n换行符无效、不能换行的解决方法
Apr 02 PHP
php结合安卓客户端实现查询交互实例
May 05 PHP
php分割合并两个字符串的函数实例
Jun 19 PHP
php根据日期或时间戳获取星座信息和生肖等信息
Oct 20 PHP
php简单日历函数
Oct 28 PHP
php+ajax实现无刷新文件上传功能(ajaxuploadfile)
Feb 11 PHP
Yii2压缩PHP中模板代码的输出问题
Aug 28 PHP
php 中phar包的使用教程详解
Oct 26 PHP
在 Laravel 6 中缓存数据库查询结果的方法
Dec 11 PHP
PHP PDO数据库操作预处理与注意事项
Mar 16 #PHP
php生成word并下载代码实例
Mar 15 #PHP
PHP-FPM的配置与优化讲解
Mar 15 #PHP
php-fpm中max_children的配置
Mar 15 #PHP
使用Zookeeper分布式部署PHP应用程序
Mar 15 #PHP
php根据命令行参数生成配置文件详解
Mar 15 #PHP
详解PHP的抽象类和抽象方法以及接口总结
Mar 15 #PHP
You might like
php根据isbn书号查询amazon网站上的图书信息的示例
2014/02/13 PHP
解密ThinkPHP3.1.2版本之独立分组功能应用
2014/06/19 PHP
php导入excel文件到mysql数据库的方法
2015/01/14 PHP
使用PHP生成图片的缩略图的方法
2015/08/18 PHP
微信公众平台DEMO(PHP)
2016/05/04 PHP
php实现与python进行socket通信的方法示例
2017/08/30 PHP
PHP实现Snowflake生成分布式唯一ID的方法示例
2020/08/30 PHP
Ajax,UTF-8还是GB2312 eval 还是execScript
2008/11/13 Javascript
jqgrid 简单学习笔记
2011/05/03 Javascript
动态加载js、css等文件跨iframe实现
2014/02/24 Javascript
jQuery插件bxSlider实现响应式焦点图
2015/04/12 Javascript
javascript中利用柯里化函数实现bind方法
2016/04/29 Javascript
JavaScript数组的定义及数字操作技巧
2016/06/06 Javascript
利用JS实现页面删除并重新排序功能
2016/12/09 Javascript
微信小程序日历组件calendar详解及实例
2017/06/08 Javascript
SVG动画vivus.js库使用小结(实例代码)
2017/09/14 Javascript
iframe与主框架跨域相互访问实现方法
2017/09/14 Javascript
不使用 JS 匿名函数理由
2017/11/17 Javascript
Vue实现左右菜单联动实现代码
2018/08/12 Javascript
使用Vue实现调用接口加载页面初始数据
2019/10/28 Javascript
vue 路由子组件created和mounted不起作用的解决方法
2019/11/05 Javascript
VUE UPLOAD 通过ACTION返回上传结果操作
2020/09/07 Javascript
JS绘图Flot应用图形绘制异常解决方案
2020/10/16 Javascript
python下调用pytesseract识别某网站验证码的实现方法
2016/06/06 Python
Python中super函数用法实例分析
2019/03/18 Python
win10环境下配置vscode python开发环境的教程详解
2019/10/16 Python
selenium+Chrome滑动验证码破解二(某某网站)
2019/12/17 Python
Python属性和内建属性实例解析
2020/01/14 Python
Mac中PyCharm配置Anaconda环境的方法
2020/03/04 Python
html5超简单的localStorage实现记住密码的功能实现
2017/09/07 HTML / CSS
瑞典网上购买现代和复古家具:Reforma
2019/10/21 全球购物
毕业生个人投资创业计划书
2014/01/04 职场文书
《锄禾》教学反思
2014/04/08 职场文书
单位活动策划方案
2014/08/17 职场文书
英语读书笔记
2015/07/02 职场文书
靠谱的活动总结
2019/04/16 职场文书