递归地将目录中的所有子目录和文件添加到数组中
Recursively adding all sub directories and files from a directory into an array
我有一个函数可以从给定目录获取所有目录和文件,但是它获取它们的方式使得使用起来有点复杂:
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)
}
}
...
}
如您所见,我希望目录中的每个文件都在目录中。我怎样才能重构这个函数以获得我想要的输出?
我会删除 "recursive" 参数并尝试这样的操作:
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;
}
关于 (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/"));
我个人喜欢保持简单,从我的个人图书馆中挑选了这一张,
可能不是最好的答案,但它完成了技巧
<?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);
一个简单但完整的示例,它使用 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);
我有一个函数可以从给定目录获取所有目录和文件,但是它获取它们的方式使得使用起来有点复杂:
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)
}
}
...
}
如您所见,我希望目录中的每个文件都在目录中。我怎样才能重构这个函数以获得我想要的输出?
我会删除 "recursive" 参数并尝试这样的操作:
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;
}
关于 (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/"));
我个人喜欢保持简单,从我的个人图书馆中挑选了这一张, 可能不是最好的答案,但它完成了技巧
<?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);
一个简单但完整的示例,它使用 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);