我正在编写一个脚本,需要检查给定目录中的任何文件最近是否被修改。我以为我可以简单地使用find
,但是即使它没有找到匹配项,它也会报告成功:
$ touch afile
$ find . -mtime -1 -iname ????*
./afile
$ echo $?
0
$ find . -mtime +1 -iname ????*
$ echo $?
0
(我只是使用 iname 谓词来排除 '.' ,但与 的行为相同-type f
)
我不想解析ls
这样的输出可能脾气暴躁。
使用test -n "$(find . -mtime -1 -iname ????*)"
似乎给出了预期的结果,但我并不觉得这是一个特别优雅的解决方案。
我不需要知道哪些文件已被修改 - 只要最近有任何文件发生更改(其中“最近”通常是 1 天)。
有一个更好的方法吗?
(搜索谷歌,我只是得到了很多关于如何执行简单查找的 SEO 废话)
答案1
是的,正如您所看到的,您不能使用 find 的退出代码来执行此类操作。退出代码仅与命令是否成功运行有关,与是否找到文件无关。毕竟,找不到文件并不是失败,而是成功:您得到了正确的结果,没有文件。这在man find
(引用自 GNU find)中进行了解释:
退出状态
如果所有文件均已成功处理,find 将以状态 0 退出;如果发生错误,则状态大于 0。这是故意一个非常广泛的描述,但如果返回值非零,则不应依赖结果的正确性寻找。
当发生某些错误时,find 可能会立即停止,而不会完成指定的所有操作。例如,某些起始点可能尚未检查或某些待处理的程序调用 -exec ... {} +或者-execdir ... {} +可能尚未执行。
你确实可以使用-n $(find ...)
,我真的没有看到那里的问题。由于文件名不能包含NUL
,我想不出有什么情况可以打破这一点。如果你想要稍微干净一点的东西,你可以尝试使用数组,然后测试数组中元素的数量是否超过0
。像这样的东西:
$ findResults=( $(find . -mtime -1 -iname '????*' | head -n1) )
$ [[ ${#findResults[@]} -gt 0 ]] && echo "Found files!" || echo "No files found!"
Found files!
$ findResults=( $(find . -mtime +1 -iname '????*' | head -n1) )
$ [[ ${#findResults[@]} -gt 0 ]] && echo "Found files!" || echo "No files found!"
No files found!
然而,坦率地说,我不确定这是否比更简单的[[ -n $(find ...) ]] && echo found || echo nope
.
答案2
使用 zsh,
if ()(($#)) **/????*(NDY1m-1); then
echo There are files or directories last modified within the last 24 hours
fi
() {body} args
定义并运行匿名函数这里的主体是算术表达式,如果(参数数量)非零则(($#))
返回 true$#
**/
任何级别(包括 0 级)的目录,extendglob 的缩写(*/)#
。注意,.
本身不被考虑,如果你想考虑它,你可以替换**/
为。{.,**/}
N
: 空球全局限定符:如果没有匹配,不要抱怨,而是将空列表传递给匿名函数。D
: dotglob,不要跳过隐藏文件。Y1
:在第一个匹配处停止作为优化m-1
:最后一次m
修改是在最后一天(24 小时),例如find
's-mtime -1
。
使用set -o extendedglob
,您可以使用?(#c4,)
而不是 来????*
匹配名称中至少有 4 个字符的文件。
答案3
if find . -name '????*' -mtime -1 | LC_ALL=C grep -q '^'; then
echo There are files last modified within the last 24 hours
fi
如果有大量匹配文件,效率可能会稍微高一些,因为grep -q
一旦读取一行输入就会退出,之后find
下次向管道写入内容时就会被杀死。
由于find
在进入管道时会缓冲其输出,因此一旦找到第一个文件,它就不会被杀死。
使用 GNU 或 FreeBSD find
,您可以执行以下操作:
if [ -n "$(find . -name '????*' -mtime -1 -print -quit)" ]...
在 NetBSD 上,相当于-exit
.使用 GNU find
,您可以替换-print
为-printf .
作为一个微小的优化。
您还可以执行以下操作:
if [ -n "$(
find . -name '????*' -mtime -1 -exec sh -c 'echo .; kill -s PIPE "$PPID"' ';'
)" ]...
wherefind
调用sh
输出某些内容并杀死其父级。
答案4
mtree
专门为此类事情而设计,例如验证充满二进制文件的目录是否未被恶意或损坏的代码覆盖,或者您想要检查一组文件是否没有意外更改的任何用例(无论是在数据或元数据)。
创建 ( )目录 ( ) 的-c
参考文件,包括摘要:my_dir.mtree
-p my_dir
-K sha256
$ sudo mtree -c -p my_dir -K sha256 > my_dir.mtree
下次您想要检查是否有任何文件已被修改时,请mtree
再次运行,仅提供-p
路径选项和您的 mtree 规范文件:
$ sudo mtree -p my_dir < my_dir.mtree
将列出任何不同的文件。