我正在尝试按日期顺序显示一组文件(最后修改)。
我已经完成了这个购买循环遍历数组并将其排序到另一个数组中,但是有没有更简单(更有效)的方法来做到这一点?
自 PHP 7.2.0 起,警告
create_function()
已被弃用。强烈建议不要依赖此功能。
为了后代,如果在接受的答案中链接的论坛帖子丢失或某些人不清楚,则需要的相关代码是:
<?php
$myarray = glob("*.*");
usort($myarray, create_function('$a,$b', 'return filemtime($a) - filemtime($b);'));
?>
在我的系统上对此进行了测试,并验证了它是否根据需要按文件 mtime 排序。我也使用了类似的方法(用 Python 编写)来确定我网站上最后更新的文件。
<?php
$items = glob('*', GLOB_NOSORT);
array_multisort(array_map('filemtime', $items), SORT_NUMERIC, SORT_DESC, $items);
自PHP 7.4起,最好的解决方案是使用带箭头函数的自定义排序:
usort($myarray, fn($a, $b) => filemtime($a) - filemtime($b));
您还可以使用 spaceship 运算符,它适用于各种比较,而不仅仅是整数比较。在这种情况下它不会有任何区别,但在所有排序操作中使用它是一个好习惯。
usort($myarray, fn($a, $b) => filemtime($a) <=> filemtime($b));
如果你想以相反的顺序排序,你可以否定条件:
usort($myarray, fn($a, $b) => -(filemtime($a) - filemtime($b)));
// or
usort($myarray, fn($a, $b) => -(filemtime($a) <=> filemtime($b)));
请注意,filemtime()
重复调用对性能不利。请应用memoization以提高性能。
这可以通过更好的性能来完成。usort()
接受的答案中的 会调用很多filemtime()
次。PHP 使用快速排序算法,其平均性能为1.39*n*lg(n)
. 该算法filemtime()
每次比较调用两次,因此我们将对 10 个目录条目进行大约 28 次调用,对 100 个条目进行 556 次调用,对 1000 个条目进行 8340 次调用等。以下代码对我来说很好,并且性能很好:
exec ( stripos ( PHP_OS, 'WIN' ) === 0 ? 'dir /B /O-D *.*' : 'ls -td1 *.*' , $myarray );
glob()
!如果您想扫描一个文件夹中的大量文件,而不需要特殊的通配符、规则集或任何exec()
,
我建议scandir()
,或者readdir()
。
glob()
慢很多,在Windows上甚至更慢。
引用: aalfinn
为什么 glob 在这个基准测试中看起来更慢?因为如果你这样写,glob 会递归到子目录
"mydir/*"
。只要确保没有任何子目录可以使 glob 快速运行。
"mydir/*.jpg"
更快,因为 glob 不会尝试获取子目录中的文件。
基准:
glob()
vsscandir()
讨论:
readdir()
vsscandir()
readdir vs scandir(stackoverflow)
readdir()
或scandir()
结合这些,以获得非常简洁的性能。PHP 7.4
usort( $myarray, function( $a, $b ) { return filemtime($a) - filemtime($b); } );
来源: https ://stackoverflow.com/a/60476123/3626361
PHP 5.3.0 及更新版本
usort($myarray, fn($a, $b) => filemtime($a) - filemtime($b));
如果你想更深入的兔子洞:
目录迭代器
https://www.php.net/manual/en/class.directoryiterator.php
https://www.php.net/manual/en/directoryiterator.construct.php(阅读评论!)
http://paulyg.github.io/blog/2014/06/03/directoryiterator-vs-filesystemiterator.html
<?php
function files_attachment_list($id, $sort_by_date = false, $allowed_extensions = ['png', 'jpg', 'jpeg', 'gif', 'doc', 'docx', 'pdf', 'zip', 'rar', '7z'])
{
if (empty($id) or !is_dir(sprintf('files/%s/', $id))) {
return false;
}
$out = [];
foreach (new DirectoryIterator(sprintf('files/%s/', $id)) as $file) {
if ($file->isFile() == false || !in_array($file->getExtension(), $allowed_extensions)) {
continue;
}
$datetime = new DateTime();
$datetime->setTimestamp($file->getMTime());
$out[] = [
'title' => $file->getFilename(),
'size' => human_filesize($file->getSize()),
'modified' => $datetime->format('Y-m-d H:i:s'),
'extension' => $file->getExtension(),
'url' => $file->getPathname()
];
}
$sort_by_date && usort($out, function ($a, $b) {
return $a['modified'] > $b['modified'];
});
return $out;
}
function human_filesize($bytes, $decimals = 2)
{
$sz = 'BKMGTP';
$factor = floor((strlen($bytes) - 1) / 3);
return sprintf("%.{$decimals}f", $bytes / pow(1024, $factor)) . @$sz[$factor];
}
// returns a file info array from path like '/files/123/*.extensions'
// extensions = 'png', 'jpg', 'jpeg', 'gif', 'doc', 'docx', 'pdf', 'zip', 'rar', '7z'
// OS specific sorting
print_r( files_attachment_list(123) );
// returns a file info array from the folder '/files/456/*.extensions'
// extensions = 'txt', 'zip'
// sorting by modified date (newest first)
print_r( files_attachment_list(456, true, ['txt','zip']) );