在bash shell中,$()
与` `(反引号)都是用来做命令替换用(command substitution)的。所谓的命令替换与我们第五章学过的变量替换差不多,都是用来重组命令行:
完成引号里的命令行,然后将其结果替换出来,再重组命令行。
例如:
$ echo the last sunday is $(date -d "last sunday" +%Y-%m-%d)
如此便可方便得到上一星期天的日期了…^_^
在操作上,用$()
或` `都无所谓,只是我”个人"比较喜欢用$()
,理由是:
-
` `很容易与' '(单引号)搞混乱,尤其对初学者来说。有时在一些奇怪的字形显示中,两种符号是一模一样的(直竖两点)。当然了,有经验的朋友还是一眼就能分辩两者。只是,若能更好的避免混乱,又何乐不为呢?^_^
-
在多层次的复合替换中,
$()
则比较直观。例如:这是错的:
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}B
BB
不过,假如你只看到${}
只能用来界定变量名称的话,那你就实在太小看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)) # 结果为42(16进位转十进制)
以一个实用的例子来看看吧:假如当前的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,这次暂时说这么多…