首先find命令的签名:
find [-H] [-L] [-P] [-D debugopts] [-Olevel] [starting-point...] [expression]
最前面的[-H] [-L] [-P]选项是控制对符号链接的处理的。[-D debugopts]是打印find命令的调试信息,[-Olevel]控制查询优化等级,find命令重新排序测试项以加速执行,同时保持整体效果;也就是说,具有副作用的副词不会相对于彼此重新排序。-Olevel 1是默认优化等级,这将重新排列表达式从而使只针对文件名的测试项(比如-name和-regex)被首先执行。搜索目录起点[starting-point...]可以有连个。
[expression]是一种查询说明,描述了如何匹配文件以及对匹配的文件进行的操作。[expression]由测试项Tests、行为项Actions、全局选项Global options、位置选项Positional options和操作符Operators组成。
1、测试项Tests返回一个真或假值,通常是基于我们所考察的文件的一些属性(usually on the basis of some property of a file we are considering. )。例如-empty测试只在当前文件是空时返回true;
2、行为Actions有副作用(在参与expression的布尔计算的同时,还在标准输出上打印一些东西)并且返回true或false(通常取决于其行为是否成功,比如成功打印?)。例如-print action在标准输出上打印当前文件的名字。
3、全局选项Global options影响tests和在命令行任意部分指定的options的操作。全局选项总是返回true。例如-depth选项使find以深度优先顺序遍历文件系统
4、位置选项Positional options只影响在其后的tests或actions。位置选项总是返回true。例如-regextype type选项就是位置选项,为出现在命令行其后面的正则表达式指定正则表达式方言|類型。改变命令行中出现在其后面的被-regex与-iregex tests所解释的正则表达式语法。要查看有哪些正则表达式类型是已知的,使用-regextype help。The Texinfo documentation解释了不同类型正则表达式之间区别的含义。
5、操作符Operators与表达式中其他项目一起出现。包含比如-o(逻辑OR)与-a(逻辑AND)。在缺少操作符的地方会假设为-a。如果整个表达式都不包含除了-prune或-print之外的行为,则对“整个表达式返回true”的所有文件执行-print。注意到-a操作符比-o有更高的优先级。这意味着find . -name afile -o -name bfile -print(最后两个选项隐含有-name bfile -a -print)对-name afile的文件不会打印,而只会对-name bfile的文件进行打印。
现在解释下面命令:
haypin@ubt ~/Logs tree .
.
├── 1.log
├── 2.log
└── dir1
└── temp.log
1 directory, 3 files
haypin@ubt ~/Logs find . -iname '*.log' -a -print
./1.log
./2.log
./dir1/temp.log
haypin@ubt ~/Logs find . -path './dir1' -prune -o -iname '*.log' -a -print
./1.log
./2.log
haypin@ubt ~/Logs find . -path './dir1' -prune -a -iname '*.log' -a -print
haypin@ubt ~/Logs find . -path './dir1' -prune -a -print
./dir1
haypin@ubt ~/Logs
haypin@ubt ~/Logs find . -path './dir1' -prune -o -iname '*.log' -a -print
注意-path 'path-ignore' -prune必须紧跟在find的搜索目录后,这里将要排除目录的-path 'path-ignore' -prune与一个匹配文件名的-iname '*.log'通过-o或连接,最后通过-a与-print连接。
-prune与-print、-delete、-exec command、-fprint file、-ls、-print0、-quit等都是行为,成功执行则返回true,执行失败返回false。
整个表达式从左向右解读,并按最短决策,对-a表达式左边已经false的直接返回false,不继续执行右边表达式,对-o表达式左边已经true的直接返回true,不继续执行右边表达式。上面-path './dir1' -prune成功排除掉./dir1目录及旗下所有文件后返回true,自然就不执行-o右边的-iname '*.log' -a -print了。
对find . -path './dir1' -a -prune -a -iname '*.log' -a -print,左边-path './dir1' -a -prune排除掉./dir1目录及旗下所有文件后返回true,继续考察-iname '*.log'时会因为已经排除了./dir1目录下所有文件而不会触发-iname '*.log',自然也不会对已经不存在的文件进行-a -print打印,对./1.log与./2.log由于不符合-path './dir1'所以最初就返回false。