Shell 反引号、$() 和 ${} 的区别

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。

在编写 Shell 脚本时建议使用KaTeX parse error: Can't use function '`' in math mode at position 59: …中,里层的反引号需要转义处理(̲`̲) ,而()则比较直观。例如下面的命令格式是错的:

command1 command2 command3``

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

command1 command2 command3``

或者换成 $()

command1 $(command2 $(command3))

(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 直接变量替换
一般情况下, v a r 与 var与 var与{var}并没有区别,但是用${ }会比较精确的界定变量名称的范围,比方说:

A="dablelv"
echo $AB

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

echo ${A}B

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

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

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

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

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

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

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

2.2.3KaTeX parse error: Expected '}', got '#' at position 2: {#̲}、{##}、KaTeX parse error: Expected '}', got 'EOF' at end of input: {%}与{%%}用于字符串模式匹配截断
可以过滤掉符合指定规则的字符串,不改变原变量。

${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

记忆的方法为:

去掉左边,在键盘上 # 在 $ 左边)

% 去掉右边(在鉴盘上 % 在 $ 右边)
一个符号是最小匹配,两个符号是最大匹配

原文地址:https://www.cnblogs.com/lidabo/p/14763831.html