1.命令替换
命令替换有两种方式:
- 反引号
- $()
反引号和$()
的作用相同,用于命令替换(command substitution),即完成引用命令的执行,将其结果替换出来,与变量替换差不多。
echo `date '--date=1 hour ago' +%Y-%m-%d-%H`
# 或
echo $(date '--date=1 hour ago' +%Y-%m-%d-%H)
输出:
2019-02-02-09
2019-02-02-09
在Shell 脚本中建议使用$()
,原因主要有:
(1)反引号与单引号外形相似,容易混淆。
(2)在多层次的复合替换中,内层反引号需要转义,而 $() 比较直观。例如下面的命令格式是错的:
echo `command1 `command2``
原本的意图是先将 command2 替换出来给 command1 处理,然后再将结果打印出来。然而,命令行替换的却是下面这两段:
`command1 `
``
正确的输入应该如下:
echo `command1 \`command2\``
# 或者换成 $()
echo $(command1 $(command2))
(3)反引号中对于反斜杠有特殊的处理,使用反协议对 Shell 特殊字符进行转义时需要两个反斜杠,而$()
中只需要使用一个反斜杠。比如下面的脚本,需要输出字符串$HOME
,而不是环境变量 HOME 的内容,在反引号中需要对$
符使用双反斜杠进行转义,$()
中只需要使用一个反斜杠。
var1=`echo \$HOME` #使用一个反斜杠无法完成对$符的转义
var2=`echo \\$HOME`
var3=$(echo \$HOME)
echo $var1
echo $var2
echo $var3
脚本执行结果:
/root
$HOME
$HOME
综上所述, 反引号是一个老的用法,$()
是新的用法,无论是在学习还是实际工作中,建议使用$()
。
2.变量替换
2.1 直接变量替换
一般情况下,$var
与${var}
并没有区别,但是用${ }会比较精确的界定变量名称的范围,比方说:
A="dablelv"
echo $AB
原本是打算先将$A
的结果替换出来,然后再补一个B字母于其后,但在命令行上,真正的结果却是只会替换变量名称为 AB 的值。若使用${}
就没问题了。
echo ${A}B
2.2 特殊变量替换
${}
除了直接替换变量内容,还有一些用于字符串变量的特殊功能。假设我们定义了一个字符串变量为:
file='/dir1/dir2/dir3/my.file.txt'
字符串提取
字符串提取可以使用${:}
与${::}
。
(1)${var:n}
若 n 为正数,n 从 0 开始,表示在变量 var 中提取第 n 个字符到末尾的所有字符。若 n 为负数,提取字符串最后面 n 的绝对值个字符,使用时在冒号后面加空格或一个算术表达式或整个 num 加上括号,如${var: -2}
、${var:1−3}
或 ${var:(-2)}
均表示提取最后两个字符。
# 提取第 1 个字符及其后面的所有字符:dir1//dir2/dir3/my.file.txt
${file:1}
# 提取最后 3 个字符,注意冒号后面添加一个空格:txt
${file: -3}
# 提取最后 3 个字符,冒号后面不需要添加空格:txt
${file:1-4}
# 提取最后 3 个字符,冒号后面不需要添加空格:txt
${file:(-3)}
(2)${var:n1:n2}
${var:n1:n2}
用于提取从下标 n1 开始后面 n2 个字符,其中下标 n1 与 n2 从 0 开始。如果长度 n2 为 0,结果为空串。
# 提取最左边的 5 个字符:/dir1
${file:0:5}
# 提取从第 5 个字符开始右边的连续 5 个字符:/dir2
${file:5:5}
字符串替换
${var/pattern/pattern}
表示将 var 字符串第一个匹配的 pattern 替换为另一个 pattern。不改变原变量。
# 将第一个 dir 替换为 path:/path1/dir2/dir3/my.file.txt
${file/dir/path}
# 将全部 dir 替换为 path:/path1/path2/path3/my.file.txt
${file//dir/path}
字符串截断
可以过滤掉符合指定规则的字符串,不改变原变量。
# 拿掉第一个 / 及其左边的字符串:dir1/dir2/dir3/my.file.txt
${file#*/}
# 拿掉最后一个 / 及其左边的字符串:my.file.txt
${file##*/}
# 拿掉最后一个 / 及其右边的字符串:/dir1/dir2/dir3
${file%/*}
# 拿掉第一个 / 及其右边的字符串:空串
${file%%/*}
记忆方法为:
# 去掉左边,在键盘上 # 在 % 左边)
% 去掉右边(在鉴盘上 % 在 # 右边)
一个符号是最小匹配,两个符号是最大匹配
关于变量替换符${}
其它的特殊用法,这里不详细展开,具体可以参考 Shell 特殊字符大全。