发表于 2018-8-28 07:29:20

Shell中命令替换$(...)与后置引用`...`的区别与联系

  一. 命令替换
  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解释,而只作为赋值时直接使用,文本输出时与""配合使用;
[*]  $()有在产生的结果还会再作进一步解释时,用与不用都可以,只是提高程序可读性之外。
  在操作上,这两者都能达到相应的效果,但是建议使用$( ),理由如下:

[*]  ``很容易与''搞混乱。
[*]  在多层次的复合替换中,``必须要额外的跳脱处理(反斜线),而$( )比较直观。
[*]  $( )的弊端是,并不是所有的类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}   返回数组第一个元素                a
  ${#A[@]}   返回数组元素总个数               4
  ${#A
[*]}   同上                         4
  ${#A}   返回第四个元素的长度,即def的长度   3
  A=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
页: [1]
查看完整版本: Shell中命令替换$(...)与后置引用`...`的区别与联系