用PHP5的DirectoryIterators遞歸掃描目錄
PHP5中增加了Iterator,一組有助于導(dǎo)航和處理等級數(shù)據(jù)結(jié)構(gòu)的現(xiàn)成接口,這是PHP5最有趣的新特性之一。
這些Iterator顯著減少了處理XML文檔樹或文件集合所需的代碼。PHP5中使用大量Iterator,包括ArrayIterator、 CachingIterator、LimitIterator、RecursiveIterator、SimpleXMLIterator和 DirectoryIterator。
通過DirectoryIterator可以迅速有效地對目錄中的文件進(jìn)行處理。在編碼過程中稍微增加一些創(chuàng)造力,DirectoryIterator還可用于遞歸處理嵌套式目錄樹。這兩個任務(wù)只需使用幾行代碼就可以完成,比“標(biāo)準(zhǔn)”處理方法有了顯著提高。
處理單級目錄首先我們從簡單的任務(wù)著手:處理一個單級目錄。輸入(或復(fù)制)以下代碼(列表A),修改目錄路徑以反映當(dāng)?shù)嘏渲茫?/P>
列表A
<?php$it = new DirectoryIterator('/tmp/mystuff');foreach($it as $file) {if (!$it->isDot()) {echo $file . 'n';}}?>在瀏覽器中查看這段代碼的輸出結(jié)果,你會在指定目錄中看到一個文件列表。這一切是如何發(fā)生的呢?DirectoryIterator提供一個預(yù)先確 定的接口來重述一個目錄的內(nèi)容;示例目標(biāo)目錄的位置后,就可以把它當(dāng)作一個標(biāo)準(zhǔn)的PHP數(shù)組來處理,每個元素代表目錄中的一個文件。注意它使用isDot ()方法分別過濾掉“.”和“..”目錄。
處理嵌套式目錄樹遞歸處理一個嵌套式目錄樹幾乎同樣簡單。在這種情況下,DirectoryIterator需要檢查它在單級目錄中遇到的每一個對象,確定其是一個文件還是目錄。如果是一個目錄,就更深入一級檢驗下一級的內(nèi)容。這聽起來似乎相當(dāng)復(fù)雜,在過去一般都需要15行以上的代碼。
但是,使用PHP5,你只需要兩個新的Iterator:RecursiveIterator和RecursiveIteratorIterator,它們組合了所有上述功能。見列表B:
列表B
<?php$it = new RecursiveDirectoryIterator('/tmp');foreach(new RecursiveIteratorIterator($it) as $file) {echo $file . 'n';}?>這時,輸入結(jié)果將列出起始目錄下的所有文件和目錄。不必說,如果需要處理某個特定目錄級下的所有文件——例如,遞歸壓縮一個目錄樹;或修改一系列嵌套文件的組/所有者許可時——使用這種遞歸內(nèi)置接口就非常方便。
現(xiàn)實應(yīng)用:打印一個目錄樹打印圖形目錄樹是目錄遞歸的一個常見應(yīng)用。利用Iterator處理這個任務(wù)十分簡單,因為Iterator類文檔中包含一個專門為這個應(yīng)用而編寫 的實例類。DirectoryTreeIterator(感謝Marcus Boerger)為前面討論的RecursiveIteratorIterator提供了其它改進(jìn),特別是在樹結(jié)構(gòu)中代表深度和位置的ASCII標(biāo)記。
列表C說明了DirectoryTreeIterator的用法。
列表C
<?php$it = new DirectoryTreeIterator('/tmp/cookbook/');foreach($it as $path) {echo $path . 'n';}?>以下是你看到的一部分輸出結(jié)果:
|-ch01| |-recipe01| | |-example01.php| | -example02.php| |-recipe02| | |-example01.php| | -example02.php| |-recipe03| | -example01.php...為更好了解這些DirectoryIterator的價值,嘗試用標(biāo)準(zhǔn)的文件和目錄函數(shù)對本教程中說明的三個應(yīng)用編碼。
