[[ ]] 是我们比较熟悉的符号了,从第一篇开始我们就一直在用,但我一直没有详细介绍它的用法,只用到了它的一小部分功能。本文详细介绍 [[ ]] 的用法。
[[ ]] 最常用的功能之一是比较字符串,这也是我们一直在用的功能。
# 匹配
% [[ abc == abc ]] && echo good
good
# = 和 == 是一样的,最好统一使用一种
% [[ abc = abc ]] && echo good
good
# 不匹配
% [[ abc != abd ]] && echo good
good
# 正则表达式匹配
% [[ abc =~ a.c ]] && echo good
good
# 前者字符序比后者小
% [[ abc < bcd ]] && echo good
good
# 前者字符序比后者大
% [[ cde > bcd ]] && echo good
good
# 没有 >= 和 <=
% [[ cde >= bcd ]] && echo good
zsh: parse error near `bcd'
除了在里边用等号、不等号之类比较外,还可以判断字符串是否为空:
% str=abc
# 判断字符串内容长度是否大于 0,等同于 (($#str))
% [[ -n $str ]] && echo good
good
% str=""
# 判断字符串是否为空,等同于 ((! $#str))
% [[ -z $str ]] && echo good
good
但这两种用法,我们都有更方便的其他实现方法,没有必要用它们。
[[ ]] 另一类很重要的功能是判断文件,比如判断某一个文件是否存在、是否是目录、是否可读等等。
判断 /bin/zsh 文件是否存在:
% [[ -e /bin/zsh ]] && echo good
good
% [[ -e /bin/zshh ]] && echo good
-e 可以替换成如下的选项,用法是一致的:
选项 | 符合条件的文件 |
---|---|
-b | 块设备文件 |
-c | 字符设备文件 |
-d | 目录 |
-e | 存在的任何文件 |
-f | 普通文件,含符号链接,不含目录、设备文件、socket、FIFO |
-g | 设置了 setgid 的文件 |
-h | 符号链接 |
-k | 设置了粘滞位(sticky bit)的文件 |
-p | FIFO 文件 |
-r | 对当前进程可读的文件 |
-s | 非空文件 |
-u | 设置了 setuid 的文件 |
-x | 对当前进程可执行的文件 |
-w | 对当前进程可写的文件 |
-L | 符号链接(同 -h) |
-O | 被当前进程的用户拥有的文件 |
-G | 被当前进程的用户组拥有的文件 |
-S | socket 文件 |
-N | atime 和 mtime 一样的文件 |
还有一个比较特殊的 -t 选项:
# $$ 是当前的进程 id
% ls /proc/$$/fd
0 1 10 11 2
% [[ -t 10 ]] && echo good
good
% [[ -t 3 ]] && echo good
-t 后要接数字(如果不是,相当于 0),判断当前进程是否打开了对应的 fd(进程默认会打开 0、1、2 这三个 fd,分别对应标准输入、标准输出和错误输出,此外每打开一个文件、管道或者网络连接,都会对应一个 fd,关掉后对应 fd 会消失)。
除了判断单个文件是否符合条件外,[[ ]] 还可以用来比较两个文件。
# file1 比 file2 新
% [[ file1 -nt file2 ]]
# file1 比 file2 旧
% [[ file1 -ot file2 ]]
# file1 和 file2 是否对应同一个文件(路径相同或者互为硬连接)
% [[ file1 -ef file2 ]]
[[ ]] 也可以用来比较数值,注意不是用等号、大于号、小于号等等比较,有一系列专门的符号。通常我们没必要用 [[ ]] 来比较数值,用 (( )) 更方便一些。
# -eq 是判断两个数值是否相等
% [[ 12 -eq 12 ]] && echo good
good
-eq 可以替换成下列符号,用法一样:
符号 | 含义 |
---|---|
-eq | 相等 |
-ne | 不相等 |
-lt | < |
-gt | > |
-le | <= |
-ge | >= |
# && 是逻辑与
% [[ a == a && b == b ]] && echo good
good
# || 是逻辑或
% [[ a == a || a == b ]] && echo good
good
# ! 是逻辑非
% [[ ! a == b ]] && echo good
good
# 可以一起用,! 优先级最高,其次 &&,再次 ||
% [[ ! a == b && b == a || b == b ]] && echo good
good
# 如果不确定优先级,可以加小括号
% [[ ((! a == b) && b == a) || b == b ]] && echo good
good
需要注意一下空格,[[ ]] 内侧和内容之间需要空格隔开,== 两边也需要空格。如果是在 zsh 中直接敲入,! 后边也要加一个空格,不然会被解析成历史命令。
除了 [[ ]] 符号,[ ] 符号(它是古老的 test 命令化身)也可以用来判断字符串、文件、数值等等,但功能没有 [[ ]] 全,只支持上边列的一部分功能(不支持 ==、=~、>、<、(、) ,并且逻辑与或的语法不一样,不能调整优先级,用起来很不方便),通常没有必要使用 [ ](如需使用,可以 man test 查看用法)。
本文详细介绍了 [[ ]] 的用法,基本覆盖全面了。