最近学习Linux Shell编程,对 () (()) [] [[]]等符号的用法还是有点分不太清楚,于是决定再梳理一下。今天先整理 () $() (()) 的用法。
1 单小括号()
1.1 子shell(命令组)
括号中的命令将会新开一个子shell顺序执行,所以括号中的变量不能够被脚本余下的部分使用。括号中多个命令之间用分号隔开,最后一个命令可以没有分号,各命令和括号之间不必有空格。
1.1.1 bash中
$ bash
 [csdn ~]$ i=1; echo $i;(let i+=5; echo $i); echo $i
 1
 6
 1
 [csdn ~]$ 
 

1.1.2 在zsh中
# csdn @ edu in ~ [16:08:09] 
 $ i=1; echo $i;(let i+=5; echo $i); echo $i
 1
 6
 1

在上面的实例中,执行命令i=1; echo $i 时变量i初始为1
执行命令(let i+=5; echo $i);时,会创建一个子shell来执行,其中变量值i变为6
最后执行命令echo $i,母shell中的变量i的值并没有继承子shell中的变化,仍然为1
1.2 用于定义并初始化数组
如:
a=(1 2 3 4 a b c d)
定义了数组 a并初始化其中的元素值。
1.3 用于条件判断
1.3.1 bash中
[csdn ~]$ i=1; if (test $i -lt 0); then; echo 'i<0'; else; echo 'i>=0'; fi
 bash: syntax error near unexpected token `;'
 [csdn ~]$ i=1;while (test $i -lt 10); do echo $i; let i++; done
 1
 2
 3
 4
 5
 6
 7
 8
 9

1.3.2 在zsh中
# csdn @ edu in ~ [19:00:25] C:1
 $ i=1; if (test $i -lt 0); then; echo 'i<0'; else; echo 'i>=0'; fi
 i>=0
# csdn @ edu in ~ [19:00:46] 
 $ i=1;while (test $i -lt 10); do echo $i; let i++; done          
 1
 2
 3
 4
 5
 6
 7
 8
 9
# csdn @ edu in ~ [19:00:52] 
 $ i=1;while (test $i < 10); do echo $i; let i++; done
 zsh: no such file or directory: 10
# csdn @ edu in ~ [19:02:11] 
 $ i=1;while (test $i != 10); do echo $i; let i++; done
 1
 2
 3
 4
 5
 6
 7
 8
 9

2 $():命令替换
与` `(反引号)相似,都是用来作命令替换的,即,先完成()(小拓号)或` `(反引号)里的命令行,然后将其标准输出结果替换出来。
2.1 在bash中
[csdn ~]$ echo $(pwd;echo '\\n'; (cd /;echo path:;pwd); echo '\\n'; pwd)
 /home/csdn \\n path: / \\n /home/csdn
 [csdn ~]$ echo `pwd;echo '\\n'; (cd /;echo path:;pwd); echo '\\n'; pwd`
 /home/csdn \n path: / \n /home/csdn
 [csdn ~]$ echo `pwd;echo "\\n"; (cd /;echo path:;pwd); echo "\\n"; pwd`
 /home/csdn \n path: / \n /home/csdn
 [csdn ~]$ echo $(pwd;echo "\\n"; (cd /;echo path:;pwd); echo "\\n"; pwd)
 /home/csdn \n path: / \n /home/csdn
 [csdn ~]$ echo $(pwd;echo -e "\\n"; (cd /;echo path:;pwd); echo -e "\\n"; pwd)
 /home/csdn path: / /home/csdn
 [csdn ~]$ 

在上面的实例中,我们先执行pwd命令显示当前目录路径:/home/csdn
然后执行命令echo '\\n'来换行
接着打开一个子shell执行命令: (cd /;echo path:;pwd); 其中命令cd /将当前目录改为/,再用pwd显示当前目录路径:/
然后我们执行命令echo '\\n'来换行
最后再用pwd显示当前目录路径:/home/csdn
但是echo '\\n'命令没有实现换行,用了-e选项还是不行。
2.2 在zsh中
# csdn @ edu in ~ [17:21:47] 
 $ echo $(pwd;echo '\\n'; (cd /;echo path:;pwd); echo '\\n'; pwd)
 /home/csdn 
  path: / 
  /home/csdn
# csdn @ edu in ~ [17:22:07] 
 $ echo `pwd;echo '\\n'; (cd /;echo path:;pwd); echo '\\n'; pwd`
 /home/csdn path: / /home/csdn

同样的命令,在zsh中执行正确,换行也生效了。
注:有些shell不支持这种使用方法,如tcsh。
3 双小括号(()):整数运算、进制转换
3.1 整数运算、比较
3.1.1 在bash中
# csdn @ edu in ~ [22:20:39] 
 $ bash
 [csdn ~]$ echo $((3+5))
 8
 [csdn ~]$ echo $((8 > 6))
 1
 [csdn ~]$ echo $((5+3 > 6))
 1
 [csdn ~]$ 

3.1.2 在zsh中
[csdn ~]$ zsh
# csdn @ edu in ~ [22:19:50] 
 $ echo $((3+5))      
 8
# csdn @ edu in ~ [22:20:11] 
 $ echo $((8 > 6))
 1
# csdn @ edu in ~ [22:20:27] 
 $ echo $((3+5 > 6))
 1
# csdn @ edu in ~ [22:20:39] 
 $ 

3.2 进制转换
$(( ))可以将其他进制转成十进制数显示出来。用法如下:
$((N#x))
其中,N为进制,x为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。
  
3.2.1 在bash中
$ bash
 [csdn ~]$ echo $((2#1000))
 8
 [csdn ~]$ echo $((8#1000))
 512
 [csdn ~]$ echo $((16#1000))
 4096
 [csdn ~]$ echo $((16#1000 - 1000))
 3096
 [csdn ~]$ echo $((16#1000 - 1000  > 2000)) 
 bash: 16#1000 - 1000  > 2000: syntax error: invalid arithmetic operator (error token is " > 2000")
 [csdn ~]$ echo $(((16#1000 - 1000)  > 2000)) 
 bash: (16#1000 - 1000)  > 2000: syntax error: operand expected (error token is " > 2000")
 [csdn ~]$ echo $(( $(16#1000 - 1000)  > 2000)) 
 bash: 16#1000: command not found
 bash:  > 2000: syntax error: operand expected (error token is " > 2000")
 [csdn ~]$ i=1000; $((16#$i-1000 > 2000))
 bash: 1: command not found
 [csdn ~]$ i=1000; $((16#${i}-1000 > 2000))
 bash: 1: command not found
[csdn ~]$ echo $((16#1000  > 2000)) 
 bash: 16#1000  > 2000: syntax error: invalid arithmetic operator (error token is " > 2000")
 [csdn ~]$ 

在bash中,可以顺利完成进制转换,进制转换和运算,但进制转换和比较不成功。
3.2.2 在zsh中
# csdn @ edu in ~ [15:08:45] C:1
 $ echo $((2#1000))
 8
# csdn @ edu in ~ [15:08:52] 
 $ echo $((8#1000))
 512
# csdn @ edu in ~ [15:09:00] 
 $ echo $((16#1000))
 4096
# csdn @ edu in ~ [15:09:06] 
 $ echo $((16#1000 - 1000))
 3096
# csdn @ edu in ~ [15:11:31] 
 $ echo $((16#1000 - 1000  > 2000)) 
 1

在zsh中,可以顺利完成进制转换,进制转换和运算,进制转换、运算和比较。
在这点上,zsh表现优于bash。
3.3 在(())中使用变量
3.3.1 在bash中
# csdn @ edu in ~ [15:24:06] 
 $ bash
 [csdn ~]$ i=5;echo $((i+9))
 14
 [csdn ~]$ i=12; echo $((8#i+9))
 bash: 8#i: value too great for base (error token is "8#i")
 [csdn ~]$ i=12; echo $((8#${i}+9))
 19
 [csdn ~]$ i=12; echo $((8#$i+9))
 19
 [csdn ~]$ i=12; echo $((8#$i+9 > 10))
 1
 [csdn ~]$ 

3.3.2 在zsh中
 csdn @ edu in ~ [15:22:38] 
 $ i=5; echo $((i+9))
 14
# csdn @ edu in ~ [15:23:00] 
 $ i=12; echo $((8#i+9))
 zsh: bad math expression: operator expected at `i+9'
# csdn @ edu in ~ [15:23:23] C:1
 $ i=12; echo $((8#${i}+9)) 
 19
# csdn @ edu in ~ [15:23:34] 
 $ i=12; echo $((8#${i}+9 > 10))
 1

在bash和zsh中,在 (( )) 中使用变量一般不需要加上$前缀,(( )) 会自动解析变量名,这使得代码更加简洁,也符合程序员的书写习惯。
但也有例外。比如:
i=12; echo $((8#i+9))
bash和zsh都无法识其中的变量i,于是我们要 用 $i 或 ${i}把变量i标记出来。
3.4 总结
shell类型  | 整数运算  | 比较  | 算术&比较  | 进制转换  | 进制转换&算术&比较  | 
bash  | √  | √  | √  | √  | ×  | 
zsh  | √  | √  | √  | √  | √  | 
3.5 注意
(( )) 只能用于整数算,不能用于小数(浮点数)或者字符串。要进行小数运算,可以使用 bc 命令。










