首页 > 编程 > PHP > 正文

SPL标准PHP类库(Standard PHP Library)

2020-03-22 19:02:57
字体:
来源:转载
供稿:网友
  • 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、splhtml' target='_blank'>classes、iteratorapply 等的帮助函数(helper functions),用于重载对应的功能。

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

    重载 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') );?>

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

    迭代器

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

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

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


    <?phpclass RecursiveFileFilterIterator extends FilterIterator {    // 满足条件的扩展名 www.it165.net    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 提供的其他的迭代器:

    RecursiveIteratorRecursiveIteratorIteratorOuterIteratorIteratorIteratorFilterIteratorRecursiveFilterIteratorParentIteratorSeekableIteratorLimitIteratorGlobIteratorCachingIteratorRecursiveCachingIteratorNoRewindIteratorAppendIteratorRecursiveIteratorIteratorInfiniteIteratorRegexIteratorRecursiveRegexIteratorEmptyIteratorRecursiveTreeIteratorArrayIterator

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

    郑重声明:本文版权归原作者所有,转载文章仅为传播更多信息之目的,如作者信息标记有误,请第一时间联系我们修改或删除,多谢。

  • 发表评论 共有条评论
    用户名: 密码:
    验证码: 匿名发表