什么是SPL?
SPL,PHP 標(biāo)準(zhǔn)庫(kù)(Standard PHP Library) ,此從 PHP 5.0 起內(nèi)置的組件和接口,并且從 PHP5.3 已逐漸的成熟。SPL 其實(shí)在所有的 PHP5 開(kāi)發(fā)環(huán)境中被內(nèi)置,同時(shí)無(wú)需任何設(shè)置。
似乎眾多的 PHP 開(kāi)發(fā)人員基本沒(méi)有使用它,甚至聞所未聞。究其原因,可以追述到它那陽(yáng)春白雪般的說(shuō)明文檔,使你忽略了「它的存在」。SPL 這塊寶石猶如鐵達(dá)尼的「海洋之心」般,被沉入海底。而現(xiàn)在它應(yīng)該被我們撈起,并將它穿戴在應(yīng)有的位置 ,而這也是這篇文章所要表述的觀點(diǎn)。
那么,SPL 提供了什么?
SPL 對(duì) PHP 引擎進(jìn)行了擴(kuò)展,例如 ArrayAccess、Countable 和 SeekableIterator 等接口,它們用于以數(shù)組形式操作對(duì)象。同時(shí),你還可以使用 RecursiveIterator、ArrayObejcts 等其他迭代器進(jìn)行數(shù)據(jù)的迭代操作。
它還內(nèi)置幾個(gè)的對(duì)象例如 Exceptions、SplObserver、Spltorage 以及 splautoloadregister、splclasses、iteratorapply 等的幫助函數(shù)(helper functions),用于重載對(duì)應(yīng)的功能。
這些工具聚合在一起就好比是把多功能的瑞士軍刀,善用它們可以從質(zhì)上提升 PHP 的代碼效率。那么,我們?nèi)绾伟l(fā)揮它的威力?
如何使用SPL?
SPL提供了一組標(biāo)準(zhǔn)數(shù)據(jù)結(jié)構(gòu):
雙向鏈表
SplDoublyLinkedList
雙鏈表是一種重要的線(xiàn)性存儲(chǔ)結(jié)構(gòu),對(duì)于雙鏈表中的每個(gè)節(jié)點(diǎn),不僅僅存儲(chǔ)自己的信息,還要保存前驅(qū)和后繼節(jié)點(diǎn)的地址。
PHP SPL中的SplDoublyLinkedList類(lèi)提供了對(duì)雙鏈表的操作。
SplDoublyLinkedList類(lèi)摘要如下:
SplDoublyLinkedList implements Iterator , ArrayAccess , Countable { public __construct ( void ) public void add ( mixed $index , mixed $newval ) //雙鏈表的頭部節(jié)點(diǎn) public mixed top ( void ) //雙鏈表的尾部節(jié)點(diǎn) public mixed bottom ( void ) //雙聯(lián)表元素的個(gè)數(shù) public int count ( void ) //檢測(cè)雙鏈表是否為空 public bool isEmpty ( void ) //當(dāng)前節(jié)點(diǎn)索引 public mixed key ( void ) //移到上條記錄 public void prev ( void ) //移到下條記錄 public void next ( void ) //當(dāng)前記錄 public mixed current ( void ) //將指針指向迭代開(kāi)始處 public void rewind ( void ) //檢查雙鏈表是否還有節(jié)點(diǎn) public bool valid ( void ) //指定index處節(jié)點(diǎn)是否存在 public bool offsetExists ( mixed $index ) //獲取指定index處節(jié)點(diǎn)值 public mixed offsetGet ( mixed $index ) //設(shè)置指定index處值 public void offsetSet ( mixed $index , mixed $newval ) //刪除指定index處節(jié)點(diǎn) public void offsetUnset ( mixed $index ) //從雙鏈表的尾部彈出元素 public mixed pop ( void ) //添加元素到雙鏈表的尾部 public void push ( mixed $value ) //序列化存儲(chǔ) public string serialize ( void ) //反序列化 public void unserialize ( string $serialized ) //設(shè)置迭代模式 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 ) }
使用起來(lái)也比較簡(jiǎn)單
$list = new SplDoublyLinkedList(); $list->push('a'); $list->push('b'); $list->push('c'); $list->push('d'); $list->unshift('top'); $list->shift(); $list->rewind();//rewind操作用于把節(jié)點(diǎn)指針指向Bottom所在的節(jié)點(diǎn) echo 'curren node:'.$list->current()."br />";//獲取當(dāng)前節(jié)點(diǎn) $list->next();//指針指向下一個(gè)節(jié)點(diǎn) echo 'next node:'.$list->current()."br />"; $list->next(); $list->next(); $list->prev();//指針指向上一個(gè)節(jié)點(diǎn) echo 'next node:'.$list->current()."br />"; if($list->current()) echo 'current node is validbr />'; else echo 'current node is invalidbr />'; if($list->valid())//如果當(dāng)前節(jié)點(diǎn)是有效節(jié)點(diǎn),valid返回true echo "valid listbr />"; 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); //堆棧,先進(jìn)后出 $stack = new SplStack();//繼承自SplDoublyLinkedList類(lèi) $stack->push("abr />"); $stack->push("bbr />"); echo $stack->pop(); echo $stack->pop(); echo $stack->offsetSet(0,'B');//堆棧的offset=0是Top所在的位置,offset=1是Top位置節(jié)點(diǎn)靠近bottom位置的相鄰節(jié)點(diǎn),以此類(lèi)推 $stack->rewind();//雙向鏈表的rewind和堆棧的rewind相反,堆棧的rewind使得當(dāng)前指針指向Top所在的位置,而雙向鏈表調(diào)用之后指向bottom所在位置 echo 'current:'.$stack->current().'br />'; $stack->next();//堆棧的next操作使指針指向靠近bottom位置的下一個(gè)節(jié)點(diǎn),而雙向鏈表是靠近top的下一個(gè)節(jié)點(diǎn) echo 'current:'.$stack->current().'br />'; echo 'br />br />'; //隊(duì)列,先進(jìn)先出 $queue = new SplQueue();//繼承自SplDoublyLinkedList類(lèi) $queue->enqueue("abr />");//插入一個(gè)節(jié)點(diǎn)到隊(duì)列里面的Top位置 $queue->enqueue("bbr />"); $queue->offsetSet(0,'A');//堆棧的offset=0是Top所在的位置,offset=1是Top位置節(jié)點(diǎn)靠近bottom位置的相鄰節(jié)點(diǎn),以此類(lèi)推 echo $queue->dequeue(); echo $queue->dequeue(); echo "br />br />";
重載 autoloader
如果你是位「教科書(shū)式的程序員」,那么你保證了解如何使用 __autoload 去代替 includes/requires 操作惰性載入對(duì)應(yīng)的類(lèi),對(duì)不?
但久之,你會(huì)發(fā)現(xiàn)你已經(jīng)陷入了困境,首先是你要保證你的類(lèi)文件必須在指定的文件路徑中,例如在 Zend 框架中你必須使用「_」來(lái)分割類(lèi)、方法名稱(chēng)(你如何解決這一問(wèn)題?)。
另外的一個(gè)問(wèn)題,就是當(dāng)項(xiàng)目變得越來(lái)越復(fù)雜, __autoload 內(nèi)的邏輯也會(huì)變得相應(yīng)的復(fù)雜。到最后,甚至你會(huì)加入異常判斷,以及將所有的載入類(lèi)的邏輯如數(shù)寫(xiě)到其中。
大家都知道「雞蛋不能放到一個(gè)籃子中」,利用 SPL 可以分離 __autoload 的載入邏輯。只需要寫(xiě)個(gè)你自己的 autoload 函數(shù),然后利用 SPL 提供的函數(shù)重載它。
例如上述 Zend 框架的問(wèn)題,你可以重載 Zend loader 對(duì)應(yīng)的方法,如果它沒(méi)有找到對(duì)應(yīng)的類(lèi),那么就使用你先前定義的函數(shù)。
?php class MyLoader { public static function doAutoload($class) { // 本模塊對(duì)應(yīng)的 autoload 操作 } } spl_autoload_register( array('MyLoader', 'doAutoload') ); ?>
正如你所見(jiàn), spl autoload register 還能以數(shù)組的形式加入多個(gè)載入邏輯。同時(shí),你還可以利用spl autoload unregister 移除已經(jīng)不再需要的載入邏輯,這功能總會(huì)用到的。
迭代器
迭代是常見(jiàn)設(shè)計(jì)模式之一,普遍應(yīng)用于一組數(shù)據(jù)中的統(tǒng)一的遍歷操作??梢院敛豢鋸埖恼f(shuō),SPL 提供了所有你需要的對(duì)應(yīng)數(shù)據(jù)類(lèi)型的迭代器。
有個(gè)非常好的案例就是遍歷目錄。常規(guī)的做法就是使用 scandir ,然后跳過(guò)「.「 和 「..」,以及其它未滿(mǎn)足條件的文件。例如你需要遍歷個(gè)某個(gè)目錄抽取其中的圖片文件,就需要判斷是否是 jpg、gif 結(jié)尾。
下面的代碼就是使用 SPL 的迭代器執(zhí)行上述遞歸尋找指定目錄中的圖片文件的例子:
?php class RecursiveFileFilterIterator extends FilterIterator { // 滿(mǎn)足條件的擴(kuò)展名 protected $ext = array('jpg','gif'); /** * 提供 $path 并生成對(duì)應(yīng)的目錄迭代器 */ public function __construct($path) { parent::__construct(new RecursiveIteratorIterator(new RecursiveDirectoryIterator($path))); } /** * 檢查文件擴(kuò)展名是否滿(mǎn)足條件 */ public function accept() { $item = $this->getInnerIterator(); if ($item->isFile() in_array(pathinfo($item->getFilename(), PATHINFO_EXTENSION), $this->ext)) { return TRUE; } } } // 實(shí)例化 foreach (new RecursiveFileFilterIterator('/path/to/something') as $item) { echo $item . PHP_EOL; } ?>
你可能會(huì)說(shuō),這不是花了更多的代碼去辦同一件事情嗎?那么,查看上面的代碼,你不是擁有了具有高度重用而且可以測(cè)試的代碼了嗎 :)
下面是 SPL 提供的其他的迭代器:
自 PHP5.3 開(kāi)始,會(huì)內(nèi)置其他更多的迭代器,我想你都可以嘗試下,或許它能改變你編寫(xiě)傳統(tǒng)代碼的習(xí)慣。
SplFixedArray
SPL 還內(nèi)置了一系列的數(shù)組操作工具,例如可以使用 SplFixedArray 實(shí)例化一個(gè)固定長(zhǎng)度的數(shù)組。那么為什么要使用它?因?yàn)樗欤踔了P(guān)系著你的工資問(wèn)題 :)
我們知道 PHP 常規(guī)的數(shù)組包含不同類(lèi)型的鍵,例如數(shù)字、字符串等,并且長(zhǎng)度是可變的。正是因?yàn)檫@些「高級(jí)功能」,PHP 以散列(hash)的方式通過(guò)鍵得到對(duì)應(yīng)的值 -- 其實(shí)這在特定情況這會(huì)造成性能問(wèn)題。
而 SplFixedArray 因?yàn)槭鞘褂霉潭ǖ臄?shù)字鍵,所以它并沒(méi)有使用散列存儲(chǔ)方式。不確切的說(shuō),甚至你可以認(rèn)為它就是個(gè) C 數(shù)組。這就是為什么 SplFixedArray 會(huì)比通常數(shù)組要快的原因(僅在 PHP5.3 中)。
那到底有多快呢,下面的組數(shù)據(jù)可以讓你窺其究竟。
如果你需要大量的數(shù)組操作,那么你可以嘗試下,相信它是值得信賴(lài)的。
數(shù)據(jù)結(jié)構(gòu)
同時(shí) SPL 還提供了些數(shù)據(jù)結(jié)構(gòu)基本類(lèi)型的實(shí)現(xiàn) 。雖然我們可以使用傳統(tǒng)的變量類(lèi)型來(lái)描述數(shù)據(jù)結(jié)構(gòu),例如用數(shù)組來(lái)描述堆棧(Strack)-- 然后使用對(duì)應(yīng)的方式 pop 和 push(arraypop()、arraypush()),但你得時(shí)刻小心,·因?yàn)楫吘顾鼈儾皇菍?zhuān)門(mén)用于描述數(shù)據(jù)結(jié)構(gòu)的 -- 一次誤操作就有可能破壞該堆棧。
而 SPL 的 SplStack 對(duì)象則嚴(yán)格以堆棧的形式描述數(shù)據(jù),并提供對(duì)應(yīng)的方法。同時(shí),這樣的代碼應(yīng)該也能理解它在操作堆棧而非某個(gè)數(shù)組,從而能讓你的同伴更好的理解相應(yīng)的代碼,并且它更快。
最后,可能上述那些慘白的例子還不足矣「誘惑你」去使用 SPL。實(shí)踐出真知,SPL 更多、更強(qiáng)大的功能需要你自己去挖掘。而它正如寶石般的慢慢雕砌,才能散發(fā)光輝。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
標(biāo)簽:吐魯番 遵義 武威 徐州 荊門(mén) 遂寧 寧夏 常州
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP標(biāo)準(zhǔn)庫(kù)(PHP SPL)詳解》,本文關(guān)鍵詞 PHP,標(biāo)準(zhǔn),庫(kù),SPL,詳解,PHP,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。