1.反引号与$()用于命令替换

反引号和$()的作用相同,用于命令替换(command substitution),即完成引用的命令的执行,将其结果替换出来,与变量替换差不多。比如:

  1. echo `date '--date=1 hour ago' +%Y-%m-%d-%H`
  2. #或者
  3. echo $(date '--date=1 hour ago' +%Y-%m-%d-%H)

输出结果是相同的:2019-02-02-09。
在编写Shell脚本时建议使用$(),原因主要有: (1)反引号与单引号外形相似,容易混淆; (2)在多层次的复合替换中,里层的反引号需要转义处理(`) ,而$()则比较直观。例如下面的命令格式是错的:

  1. command1 `command2 `command3``

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

  1. command1 `command2 \`command3\``
  2. #或者换成$()
  3. command1 $(command2 $(command3))

(3)反引号中对于反斜杠有特殊的处理,使用反协议对Shell特殊字符进行转义时需要两个反斜杠,而$()中只需要使用一个反斜杠。比如下面的脚本,需要输出$HOME,而不是环境变量HOME的内容,在反引号中需要对$符使用双反斜杠进行转义,$()中只需要使用一个反斜杠。

  1. var1=`echo \$HOME` #使用一个反斜杠无法完成对$符的转义
  2. var2=`echo \\$HOME`
  3. var3=$(echo \$HOME)
  4. echo $var1
  5. echo $var2
  6. echo $var3

脚本执行结果:

  1. /root
  2. $HOME
  3. $HOME

综上所述, 反引号是一个老的用法,$()是新的用法,无论是在学习还是实际工作中,建议使用$()。

2.${}用于变量替换

2.1直接变量替换

一般情况下,$var与${var}并没有区别,但是用${ }会比较精确的界定变量名称的范围,比方说:

  1. A="dablelv"
  2. echo $AB

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

  1. echo ${A}B

2.2特殊变量替换

${} 除了直接替换变量内容,还有一些用于字符串变量的特殊功能。假设我们定义了一个字符串变量为:

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

2.2.1 ${:}与${::}用于字符串提取

字符串提取可以使用${:}与${::}。 (1)${var:n} 若n为正数,n从0开始,表示在变量var中提取第n个字符到末尾的所有字符。若n为负数,提取字符串最后面n的绝对值个字符,使用时在冒号后面加空格或一个算术表达式或整个num加上括号,如${var: -2}、${var:1−3}或 ${var:(-2)}均表示提取最后两个字符。

  1. ${file:1}:提取第1个字符及其后面的所有字符:dir1//dir2/dir3/my.file.txt
  2. ${file: -3}:提取最后3个字符,注意冒号后面添加一个空格:txt
  3. ${file:1-4}:提取最后3个字符,冒号后面不需要添加空格:txt
  4. ${file:(-3)}:提取最后3个字符,冒号后面不需要添加空格:txt

(2)${var:n1:n2}${var:n1:n2}用于提取从下标n1开始后面n2个字符,其中下标n1与n2从0开始。

  1. ${file:0:5}:提取最左边的5个字符:/dir1
  2. ${file:5:5}:提取从第5个字符开始右边的连续5个字符:/dir2

2.2.2 ${/}与${//}用于字符串模式匹配替换

${var/pattern/pattern}表示将var字符串的第一个匹配的pattern替换为另一个pattern。不改变原变量。

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

2.2.3${#}、${##}、${%}与${%%}用于字符串模式匹配截断

可以过滤掉符合指定规则的字符串,不改变原变量。

  1. ${file#*/}:拿掉第一个 / 及其左边的字符串:dir1/dir2/dir3/my.file.txt
  2. ${file##*/}:拿掉最后一个 / 及其左边的字符串:my.file.txt
  3. ${file#*.}:拿掉第一个 . 及其左边的字符串:file.txt
  4. ${file##*.}:拿掉最后一个 . 及其左边的字符串:txt
  5. ${file%/*}:拿掉最后一个 / 及其右边的字符串:/dir1/dir2/dir3
  6. ${file%%/*}:拿掉第一个 / 及其右边的字符串:(空值)
  7. ${file%.*}:拿掉最后一个 . 及其右边的字符串:/dir1/dir2/dir3/my.file
  8. ${file%%.*}:拿掉第一个 . 及其右边的字符串:/dir1/dir2/dir3/my

记忆的方法为:

  1. #是去掉左边(在鉴盘上#在$之左边)
  2. %是去掉右边(在鉴盘上%在$之右边)
  3. 一个符号是最小匹配,两个符号是最大匹配。

关于变量替换符${}其它的特殊用法,这里不详细展开,具体可以参考Shell特殊字符

https://cloud.tencent.com/developer/article/1398436