(())与( ) 还有有${ } 差在哪???

来源:互联网 发布:光明大陆 知乎 编辑:程序博客网 时间:2024/04/30 05:30

在bash shell中,$() 与` `(反引号)都是用来做命令替换用(command substitution)的。所谓的命令替换与我们第五章学过的变量替换差不多,都是用来重组命令行:

完成引号里的命令行,然后将其结果替换出来,再重组命令行。

例如:

$ echo the last sunday is $(date -d "last sunday" +%Y-%m-%d)

如此便可方便得到上一星期天的日期了…^_^

在操作上,用$()或` `都无所谓,只是我”个人"比较喜欢用$(),理由是:

  1. ` `很容易与' '(单引号)搞混乱,尤其对初学者来说。有时在一些奇怪的字形显示中,两种符号是一模一样的(直竖两点)。当然了,有经验的朋友还是一眼就能分辩两者。只是,若能更好的避免混乱,又何乐不为呢?^_^

  2. 在多层次的复合替换中, 须要额外的跳脱(\`)处理,而$() 则比较直观。例如:这是错的:

command1 `command2 `command3` `

原本的意图是要在command2 `command3`先将command3提换出来给command2处理,然后再将结果传给command1 `command2…`来处理。然而,真正的结果在命令行中却是分成了`command2`与` `两段。正确的输入应该如下:

command1 `command2 \`command3\` `

要不然,换成$()就没问题了:

command1 $(command2 $(command3))

只要你喜欢,做多少层的替换都没问题啦~~~ ^_^

不过,$()并不是没有斃端的…

首先,`基本上可用在全部的unix shell中使用,若写成shell script,其移植性比较高。而$()并不见的每一种shell都能使用,我只能跟你说,若你用bash2`的话,肯定没问题…^_^

接下来,再让我们看${}吧…它其实就是用来作变量替换用的啦。一般情况下,$var${var}并没有啥不一样。但是用${}会比较精确的界定变量名称的范围,比方说:

$ A=B$ echo $AB

原本是打算先将$A的结果替换出来,然后再补一个B字母于其后,但在命令行上,真正的结果却是只会提换变量名称为AB的值出来…若使用${}就没问题了:

$ echo ${A}BBB

不过,假如你只看到${}只能用来界定变量名称的话,那你就实在太小看bash了﹗为了完整起见,我这里再用一些例子加以说明${}的一些特异功能。

假设我们定义了一个变量为:

file=/dir1/dir2/dir3/my.file.txt

我们可以用${}分别替换获得不同的值:

${file#*/} # 拿掉第一条/及其左边的字串:dir1/dir2/dir3/my.file.txt${file##*/} # 拿掉最后一条/及其左边的字串:my.file.txt${file#*.} # 拿掉第一个.及其左边的字串:file.txt${file##*.} # 拿掉最后一个.及其左边的字串:txt${file%/*} # 拿掉最后条/及其右边的字串:/dir1/dir2/dir3${file%%/*} # 拿掉第一条/及其右边的字串:(空值)${file%.*} # 拿掉最后一个.及其右边的字串:/dir1/dir2/dir3/my.file${file%%.*} #拿掉第一个.及其右边的字串:/dir1/dir2/dir3/my

记忆的方法为:

# 是去掉左边(在键盘上#在$之左边)% 是去掉右边(在键盘上%在$之右边)单一符号是最小匹配,两个符号是最大匹配。
${file:0:5} # 提取最左边的5个字节:/dir1${file:5:5} # 提取第5个字节右边的连续5个字节:/dir2

我们也可以对变量值里的字串作替换:

${file/dir/path} # 将第一个dir提换为path:/path1/dir2/dir3/my.file.txt${file//dir/path} # 将全部dir提换为path:/path1/path2/path3/my.file.txt

利用${}还可针对不同的变数状态赋值(没设定、空值、非空值)

${file-my.file.txt}    # 假如$file没有设定,则使用my.file.txt作传回值。(空值及非空值时不作处理)${file:-my.file.txt}   # 假如$file没有设定或为空值,则使用my.file.txt作传回值。(非空值时不作处理)${file+my.file.txt}    # 假如$file设为空值或非空值,均使用my.file.txt作传回值。(没设定时不作处理)${file:+my.file.txt}   # 若$file为非空值,则使用my.file.txt作传回值。(没设定及空值时不作处理)${file=my.file.txt}    # 若$file没设定,则使用my.file.txt作传回值,同时将$file赋值为my.file.txt。(空值及非空值时不作处理)${file:=my.file.txt}   # 若$file没设定或为空值,则使用my.file.txt作传回值,同时将$file赋值为my.file.txt。(非空值时不作处理)${file?my.file.txt}    # 若$file没设定,则将my.file.txt输出至STDERR。(空值及非空值时不作处理)${file:?my.file.txt}   # 若$file没设定或为空值,则将my.file.txt输出至STDERR。(非空值时不作处理)

Tips:

以上的理解在于,你一定要分清楚unset与null及non-null这三种赋值状态。一般而言,: 与null有关,若不带 : 的话,null不受影响,若带 : 则连null也受影响。

还有哦,${#var}可计算出变量值的长度:

${#file} # 可得到27,因为/dir1/dir2/dir3/my.file.txt刚好是27个字节…

接下来,再为大家介稍一下bash的数组(array)处理方法。

一般而言,A="a b c def"这样的变量只是将$A替换为一个单一的字串,但是改为A=(a b c def),则是将$A定义为数组…bash的数组替换方法可参考如下方法:

${A[@]}或${A[*]} # 可得到a b c def(全部组数)${A[0]} # 可得到a(第一个组数),${A[1]}则为第二个组数…${#A[@]}或${#A[*]} #可得到4(全部组数数量)${#A[0]} #可得到1(即第一个组数(a)的长度),${#A[3]}可得到3(第四个组数(def)的长度)A[3]=xyz # 则是将第四个组数重新定义为xyz…

能够善用bash的$()${}可大大提高及简化shell在变量上的处理能力哦~~~ ^_^

好了,最后为大家介绍$(())的用途吧:它是用来作整数运算的。在bash中,$(())的整数运算符号大致有这些:+ - * /:分别为“加、减、乘、除”。%:余数运算。& | ^!:分别为“AND、OR、XOR、NOT”运算。

例:

$ a=5;b=7;c=2$ echo $((a+b*c))19$ echo $(((a+b)/c))6$ echo $(((a*b)%c))1

$(())中的变量名称,可于其前面加$符号来替换,也可以不用,如:
$(($a + $b * $c))也可得到19的结果

此外,$(())还可作不同进位(如二进位、八进位、十六进位)作运算呢,只是,输出结果皆为十进制而已:

echo $((16#2a)) # 结果为4216进位转十进制)

以一个实用的例子来看看吧:假如当前的umask是022,那么新建文件的权限即为:
umask022 echo "obase=8;You can't use 'macro parameter character #' in math mode(umask))))" | bc
644

事实上,单纯用(())也可重定义变量值,或作testing:

a=5;((a++)) #可将$a重定义为6
a=5;((a--)) 则为a=4
a=5;b=7;((a < b)) 会得到0(true)的返回值。

常见的用于(())的测试符号有如下这些:
<:小于
\>:大于
<=:小于或等于
\>=:大于或等于
==:等于
!=:不等于

不过,使用(())作整数测试时,请不要跟[ ]的整数测试搞混乱了。(更多的测试我将于第十章为大家介绍)怎样?好玩吧..^_^ okay,这次暂时说这么多…