一. 命令替换
a.在bash中,$( )与``(反引号)都是用来作命令替换的。
b.命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行。
例子1:
$ echo today is $(date "+%Y-%m-%d") today is 2014-07-01
二. $( )与``(反引号)
(1)区别
-
- `CMD`在执行时,shell会不管``内是什么都先进行解释,再把解释后的最终结果送给shell去执行.如果解释后的最终结果不是shell可执行的命令时,则会出错.当然,仅仅为了把CMD执行后的内容作为文本输出,则没什么问题了啦.
- $(CMD)在执行时,如果CMD是命令,则直接丢给shell去执行;如果是变量取值,则也仅作第一层的字面解释后丢给shell去执行.
(2)使用场合
-
- ``用在产生的结果不会再送给shell解释,而只作为赋值时直接使用,文本输出时与""配合使用;
- $()有在产生的结果还会再作进一步解释时,用与不用都可以,只是提高程序可读性之外。
(3)在操作上,这两者都能达到相应的效果,但是建议使用$( ),理由如下:
-
- ``很容易与‘‘搞混乱。
- 在多层次的复合替换中,``必须要额外的跳脱处理(反斜线),而$( )比较直观。
- $( )的弊端是,并不是所有的类unix系统都支持这种方式,反引号基本上可用在全部的 unix shell 中使用,若写成 shell script ,其移植性比较高。
- ``(反引号)需要进行转义。而$( )无需转义。
例子2:
# 将cmd1执行结果作为cmd2参数,再将cmd2结果作为cmd3的参数 cmd3 $(cmd2 $(cmd1)) # 如果是用反引号,直接引用是不行的,还需要作跳脱处理 cmd3 `cmd2 \`cmd1\``
例子3:
echo `echo ‘\\‘` #输出: echo $(echo ‘\\‘) #输出 \\
例子4:
假设:
VS100COMNTOOLS=C:\Program Files\Microsoft Visual Studio 10.0\Common7\Tools\
在终端输入:
echo `echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\.\+#\1#‘`
输出
C:\Program Files\Microsoft Visual Studio 10.0\Common7\Tools\
echo $(echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\.\+#\1#‘)
输出
C:\Program Files\Microsoft Visual Studio 10.0\Common7
$(...)形式的命令替换达到了预期目的,而`...`却没有达到的原因是:
`...`将sed ‘...\\...‘中的\\转义成了sed ‘...\....‘
改成
echo `echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\\\.\+#\1#‘`
或者使用$(...)即可
三. 拓展
(1)${ }变量替换
一般情况下,$var与${var}是没有区别的,但是用${ }会比较精确的界定变量名称的范围
$ A=B $ echo ${A}B #输出: BB
(2)取路径、文件名、后缀
先赋值一个变量为一个路径,如: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
记忆方法如下:
- # 是去掉左边(在键盘上 # 在 $ 之左边)
- % 是去掉右边(在键盘上 % 在 $ 之右边)
- 单一符号是最小匹配;两个符号是最大匹配
- *是用来匹配不要的字符,也就是想要去掉的那部分
- 还有指定字符分隔号,与*配合,决定取哪部分
(3)取子串及替换
命令 解释 结果
${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} 获取变量长度 27
(4)根据状态为变量赋值
命令 解释 备注
${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=txt} 若 $file 没设定,则回传 txt ,并将 $file 赋值为 txt 空值及非空值不作处理
${file:=txt} 若 $file 没设定或空值,则回传 txt ,将 $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 也受影响.
(5)数组
A="a b c def" # 定义字符串 A=(a b c def) # 定义字符数组
命令 解释 结果
${A[@]} 返回数组全部元素 a b c def
${A[*]} 同上 a b c def
${A[0]} 返回数组第一个元素 a
${#A[@]} 返回数组元素总个数 4
${#A[*]} 同上 4
${#A[3]} 返回第四个元素的长度,即def的长度 3
A[3]=xyz 则是将第四个组数重新定义为 xyz
(6)$(( ))与整数运算
bash中整数运算符号
符号 功能
+ - * / 分别为加、减、乘、除
% 余数运算
& | ^ ! 分别为“AND、OR、XOR、NOT”
在 $(( )) 中的变量名称,可于其前面加 $ 符号来替换,也可以不用。
$ a=5;b=7;c=2 $ echo $((a+b*c)) 19 $ echo $(($a+$b*$c)) 19
(7)进制转换
$(( ))可以将其他进制转成十进制数显示出来。用法如下:
echo $((N#xx))
其中,N为进制,xx为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。
$ echo $((2#110)) # 二进制转十进制 6 $ echo $((16#2a)) # 十六进制转十进制 42 $ echo $((8#11)) # 八进制转十进制 9
(7)(( ))重定义变量值
$ a=5;b=7 $ ((a++));echo $a 6 $ ((a--));echo $a 5 $ ((a<b));echo $? 0
使用(( ))作整数测试时,不要跟[ ]的整数测试搞混乱了。