之前看过(老男孩)前辈的一篇文章《》,近日终于有机会得以验证,但与老男孩前辈的文章中具体讲的有些不太一样,故而专门写一篇文章重新整理这个问题。虽然是同一个话题,和同一种思路,但细节上有一定的差别。

需求:

删除一个目录下的所有文件,但保留一个指定文件。

场景与要求:

  1. 目录下有目录和文件,目录下有子目录和子文件(这是通常的场景)

  2. 目录下有符号链接文件、隐藏文件等

  3. 如果是当前目录,则需要排除当前目录

  4. 执行成功则返回代码0,并且不提示任何警告和错误

解决办法:

例如删除当前目录下的所有文件,除了deploy.sh(原始场景:deploy.sh完成部署工作,但不需要此项目时,执行destroy函数,删除当前目录下的所有文件,除了deploy.sh文件本身)

简单验证,可以如此执行:

find -L ./ -maxdepth 1 ! -name "deploy.sh" ! -wholename "./"

如果是真的要删除,则是(也是此文主要想表达的方式):

find -L ./ -maxdepth 1 ! -name "deploy.sh" ! -wholename "./" -exec rm -rf {} \;

如果放到脚本中执行,则可以(其中$WORKDIR是脚本工作目录):

find -L $WORKDIR -maxdepth 1 ! -name "$(basename $0)" ! -wholename "$WORKDIR"  -exec rm -rf {} \;

原理或解释:

主要是利用find命令结合rm命令,

  • -L 表示find时包含符号链接文件

  • -maxdepth 1 表示查找目录深度为1,也就是只处理目录下的目录和文件,其他的目录中的文件或目录一概不做处理,如果不指定此参数,很有可能遭遇“No such file or directory”问题,原因是,find出来的内容是无法经过排序的,有时会造成父目录已经被rm删除,子目录的路径信息还会传递给rm,造成rm无法找到文件或目录,从而报错。

  • ! -name "$(basename $0)"  表示不包含当前脚本名字的文件

  • ! -wholename "$WORKDIR"  表示不包含脚本或命令所在的当前工作目录,之所以用-wholename而不是-name是因为兼容某些Unix系统,如果仅用-name会得到一条警告“find: warning: Unix filenames usually don't contain slashes (though pathnames do).  That means that '-name `./deploy.sh'' will probably evaluate to false all the time on this system.  You might find the '-wholename' test more useful, or perhaps '-samefile'.  Alternatively, if you are using GNU grep, you could use 'find ... -print0 | grep -FzZ `./deploy.sh''.”。

tag:find和rm删除,find删除文件,find符号链接文件,find maxdepth用途,find删除

--end--