Answers:
我最短的方法使用zsh:
print -rl -- **/*(.Om)
(D
如果您还想列出隐藏文件或隐藏目录中的文件,请添加glob限定符)。
如果您有GNU查找,请使其打印文件修改时间并以此排序。我假设文件名中没有换行符。
find . -type f -printf '%T@ %p\n' | sort -k 1 -n | sed 's/^[^ ]* //'
如果您有Perl(同样,假设文件名中没有换行符):
find . -type f -print |
perl -l -ne '
$_{$_} = -M; # store file age (mtime - now)
END {
$,="\n";
print sort {$_{$b} <=> $_{$a}} keys %_; # print by decreasing age
}'
如果您使用Python(同样,假设文件名中没有换行符):
find . -type f -print |
python -c 'import os, sys; times = {}
for f in sys.stdin.readlines(): f = f[0:-1]; times[f] = os.stat(f).st_mtime
for f in sorted(times.iterkeys(), key=lambda f:times[f]): print f'
如果您具有对该服务器的SSH访问权限,请在功能更好的计算机上通过sshfs挂载目录:
mkdir mnt
sshfs server:/path/to/directory mnt
zsh -c 'cd mnt && print -rl **/*(.Om)'
fusermount -u mnt
仅使用POSIX工具,要复杂得多,因为没有找到文件修改时间的好方法。检索文件时间的唯一标准方法是ls
,并且输出格式取决于语言环境且难以解析。
如果您可以写入文件,并且只关心常规文件,并且文件名中没有换行符,那么这很可怕:在单个目录中创建指向所有文件的硬链接,然后按修改时间对它们进行排序。
set -ef # disable globbing
IFS='
' # split $(foo) only at newlines
set -- $(find . -type f) # set positional arguments to the file names
mkdir links.tmp
cd links.tmp
i=0 list=
for f; do # hard link the files to links.tmp/0, links.tmp/1, …
ln "../$f" $i
i=$(($i+1))
done
set +f
for f in $(ls -t [0-9]*); do # for each file, in reverse mtime order:
eval 'list="${'$i'} # prepend the file name to $list
$list"'
done
printf %s "$list" # print the output
rm -f [0-9]* # clean up
cd ..
rmdir links.tmp
find . -print | xargs -n99999 -s999999 ls -ltr
。但这有一个问题:(1)xargs
可能不允许-m
大于512或-s
大于5120,并且(b)即使您可以解决这个问题,组合参数列表和环境仍然存在内核施加的最大大小。您的大多数想法(保存Perl和Python的想法)都存在相同的问题,这就是为什么我特别避免构建长命令行的原因。
zsh
在一般情况下递归水珠。
print -rl **/*
,唯一的限制是您有多少可用内存),还有更多zargs
。find … | xargs … ls
如果xargs
最终ls
只调用一次,您的提议将正确排序,如果文件名中包含特殊字符,则该提议将不起作用。
假设GNU find
:
find . -printf '%T@ %c %p\n' | sort -k 1n,1 -k 7 | cut -d' ' -f2-
如果要首先列出最新的文件,请更改1n,1
为1nr,1
。
如果您没有GNU,find
它将变得更加困难,因为ls
的时间戳格式变化很大(例如,最近修改的文件具有不同的时间戳风格)。
ls --time-style="..."
-我不确定这是标准格式还是GNU ls,可能是GNU。
ls
没有该选项。
一个人可以尝试一下(虽然必须自己构建)https://github.com/shadkam/recentmost
很好地处理文件名中的空格-并非您应该使用它们!
$ find . -type f -not -path '*/\.*' -printf '%TY.%Tm.%Td %THh%TM %Ta %p\n' |sort -nr |head -n 10
2017.01.25 18h23 Wed ./indenting/Shifting blocks visually.mht
2016.12.11 12h33 Sun ./tabs/Converting tabs to spaces.mht
2016.12.02 01h46 Fri ./advocacy/2016.Vim or Emacs - Which text editor do you prefer?.mht
2016.11.09 17h05 Wed ./Word count - Vim Tips Wiki.mht
一般而言,按日期查找文件(这对于原始海报不起作用,但我到这里来了,所以我认为其他人也可能如此)。在我的用例中,我想列出文件,以便在删除之前对其进行检查。
使用findutils 4.6.0,我喜欢:
find . -type f -mtime +270 -exec ls -laFGht --time-style=iso {} \+
上述命令查找文件(-type f
)中,在当前工作目录(.
),其被修改超过270天前(-mtime +270
另外-mtime 0
将产生的最后24小时,-mtime -5
示出了最后5天)。然后ls
按日期列出它们,最新的第一(-exec ls -laFGht --time-style=iso {} \+
)
这是输出示例:
-rwxrwx---+ 1 user1 208M 2018-07-16 ./filename.bak*
-rwxrwx---+ 1 user1 702 2018-07-15 ./filename.ldf*
-rwxrwx---+ 1 user1 208M 2018-07-15 ./filename.bak*
很棒的事情是,一旦列表被检查过,用find
delete命令替换列表部分就很简单了:
find . -type f -mtime +270 -delete