我有一个函数,可以从给定目录中获取所有目录和文件,但是获取它们的方式使使用它变得更加复杂:
function getFileLists($dir, $recursive=FALSE) {
// retrieve an array of all the directories and files in a certain directory
$retval = [];
if (substr($dir, -1) !== "/") {
$dir .= "/";
}
$d = @dir($dir) or die("unable to open {$dir} for reading, permissions?");
while(FALSE !== ($entry = $d->read())) {
if ($entry{0} === ".") { continue; }
if (is_dir("{$dir}{$entry}")) {
$retval[] = [
'name' => "{$dir}{$entry}",
'last_modified' => filemtime("{$dir}{$entry}")
];
if($recursive && is_readable("{$dir}{$entry}/")) {
$retval = array_merge($retval, getFileLists("{$dir}{$entry}/", TRUE));
}
} elseif (is_readable("{$dir}{$entry}")) {
$retval[] = [
'name' => "{$dir}{$entry}",
'last_modified' => filemtime("{$dir}{$entry}")
];
}
}
$d->close();
return $retval;
}
在给定目录上运行它时,将产生以下结果:
array(14) {
[0]=>
array(2) {
["name"]=>
string(15) "./kb_data/admin"
["last_modified"]=>
int(1543591247)
}
[1]=>
array(2) {
["name"]=>
string(28) "./kb_data/admin/testfile.txt"
["last_modified"]=>
int(1543591238)
}
...
}
这很棒,但是这使我很难将其显示为tree
。我实际上想要做的是按照以下方式生产东西:
array(14) {
[0]=>
array(3) {
["name"]=>
string(16) "./kb_data/shared"
["last_modified"]=>
int(1543591258)
["files"] => array(#) {
["name"]=>
string(29) "./kb_data/shared/testfile.txt"
["last_modified"]=>
int(1543591238)
}
}
...
}
如您所见,我希望目录中的每个文件都在目录内。我该如何重构该函数以获得所需的输出?
答案 0 :(得分:3)
(SPL)目录迭代器怎么样
function getFileLists($dir) {
$rec = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($dir), RecursiveIteratorIterator::SELF_FIRST);
$ar = array();
foreach ($rec as $splFileInfo) {
$path = $splFileInfo->isDir()
? array($splFileInfo->getFilename() => array())
: array($splFileInfo->getFilename());
for ($depth = $rec->getDepth() - 1; $depth >= 0; $depth--) {
$path = array($rec->getSubIterator($depth)->current()->getFilename() => $path);
}
$ar = array_merge_recursive($ar, $path);
}
return $ar;
}
print "<pre>";
print_r(getFileLists("/Library/WebServer/Documents/am-web/"));
答案 1 :(得分:1)
我个人喜欢保持简单,将其从我的个人书架中拔出, 可能不是最佳答案,但可以解决问题
<?php
function glob_recursive($dir, $recursive = false) {
$files = [];
foreach (glob($dir . DIRECTORY_SEPARATOR . '*') as $object) {
if (in_array($object, ['.', '..'])) {
continue;
}
$file = [];
$file['name'] = $object;
if (false !== $recursive && is_dir($object)) {
$file['files'] = glob_recursive($object);
}
$files[] = $file;
}
return $files;
}
$dir = __DIR__;
$list = glob_recursive($dir);
答案 2 :(得分:1)
一个简单但完整的示例,该示例使用DirectoryIterator
列出目录中的所有文件并进行递归,以避免必须处理递归迭代器数据以提供所需的所有信息。我喜欢DirectoryIterator
的地方在于,它提供了一个对象,您可以在其中提取许多文件的详细信息(例如,使用getMTime()
的上次修改时间)。
function buildTree ( $path ) {
$paths = [];
foreach (new DirectoryIterator ($path) as $file) {
if ( !$file->isDot() ) {
$newFile = ["name" => $file->getRealPath(),
"last_modified" => $file->getMTime()];
if ($file->isDir()) {
$newFile["files"] = buildTree($file->getRealPath());
}
$paths[] = $newFile;
}
}
return $paths;
}
$paths = buildTree($root);
答案 3 :(得分:-1)
我将删除“递归”参数,然后尝试执行以下操作:
function getFileLists($dir)
{
echo "Processing $dir\n";
// retrieve an array of all the directories and files in a certain directory
$retval = [];
if (substr($dir, -1) !== "/") {
$dir .= "/";
}
$d = @dir($dir) or die("unable to open {$dir} for reading, permissions?");
while (false !== ($entry = $d->read())) {
if ($entry{0} === ".") {
continue;
}
if (is_dir("{$dir}{$entry}")) {
$retval[] = [
'name' => "{$dir}{$entry}",
'last_modified' => filemtime("{$dir}{$entry}"),
//recursive call
'files' => array_merge($retval, getFileLists("{$dir}{$entry}/"))
];
} elseif (is_readable("{$dir}{$entry}")) {
$retval[] = [
'name' => "{$dir}{$entry}",
'last_modified' => filemtime("{$dir}{$entry}")
];
}
}
$d->close();
return $retval;
}