一、反引号【 `` 】和 $() 的区别与联系
‘$‘(参数替换)和‘`‘(命令替换)
反引号【 `` 】和 $() 的用法是一样的。在执行一条命令时,会先将其中的 ``,或者是$() 中的语句当作命令执行一遍,再将结果加入到原命令中重新执行
例如:
echo `ls` #会先执行 ls 得到xx.sh等,再替换原命令为:
echo xx.sh
最后执行结果为: xx.sh
平时我们遇到的把一堆命令的执行结果输出到一个变量中,需要用这个命令替换符括起来,但是根据POSIX规范,要求系统工程师采用的是 $(命令)的形式。所以,我们最好还是遵循这个规范,少用``,多用$()。
二、[[...]] 与 [...] 的区别与联系
1、[...]
1) 单中括号 " [ "是一个可执行程序,路径是"/usr/bin/[",是bash的内部命令。它与可执行程序"test"是等价的。
如果我们不用绝对路径指明,通常我们用的都是bash自带的命令。if/test结构中的左中括号是调用test的命令标识,右中括号是关闭条件判断的。这个命令把它的参数作为比较表达式或者作为文件测试,并且根据比较的结果来返回一个退出状态码。if/test结构中并不是必须右中括号,但是新版的Bash中要求必须这样。
2) Test和[]中可用的比较运算符只有 == 和 != ,两者都是用于字符串比较的,不可用于整数比较,整数比较只能使用-eq,-gt这种形式。无论是字符串比较还是整数比较都不支持大于号小于号。如果实在想用,对于字符串比较可以使用转义形式,如果比较"ab"和"bc":[ ab \< bc ],结果为真,也就是返回状态为0。[ ]中的逻辑与和逻辑或使用-a 和-o 表示。
3) 字符范围。用作正则表达式的一部分,描述一个匹配的字符范围。作为test用途的中括号内不能使用正则。
4) 在一个array 结构的上下文中,中括号用来引用数组中每个元素的编号。
2、[ [... ...] ]
1) 双中括号 " [[ "是脚本解释程序(bash,sh等)的关键字,并不是一个命令。[[ ]] 结构比[ ]结构更加通用。在[[和]]之间所有的字符都不会发生文件名扩展或者单词分割,但是会发生参数扩展和命令替换。
2) 支持字符串的模式匹配,使用=~操作符时甚至支持shell的正则表达式。字符串比较时可以把右边的作为一个模式,而不仅仅是一个字符串,比如[[ hello == hell? ]],结果为真。[[ ]] 中匹配字符串或通配符,不需要引号。
3) 使用[[ ... ]]条件判断结构,而不是[ ... ],能够防止脚本中的许多逻辑错误。比如,&&、||、<和> 操作符能够正常存在于[[ ]]条件判断结构中,但是如果出现在[ ]结构中的话,会报错。
4) bash把双中括号中的表达式看作一个单独的元素,并返回一个退出状态码。
三、eval 的使用用法
如果变量中包含任何需要shell直接在命令行中看到的字符(不是替换的结果),就可以使用eval。命令行结束符(; |&),I/o重定向符(< >)和引号就属于对shell具有特殊意义的符号,必须直接出现在命令行中。
1. eval command-line
eval主要用在对参数的特殊处理上面的,一般的命令行,shell处理参数就只执行一遍,像转义和变量转变;但加上eval后就可以对参数经行两遍处理。一个eval只能使shell对参数多一次处理,因此有几个eval就可以多加几次,即eval eval command-line 这样就能对参数进行三次编译,但此时应特别注意参数的转义,下面有例子说明。
eval最常见的用法是将动态生成的命令行计算并执行。 eval命令会计算(evalue)它的参数,这些参数作为表达式计算后重新组合为一个字符串,然后作为一个命令被执行。
pipe="|" eval ls $pipe wc -l
shell第1次扫描命令行时,它替换出pipe的值|,接着eval使它再次扫描命令行,这时shell把|作为管道符号了。
2. eval echo \$$# 取得最后一个参数
(1)
第一遍扫描后,shell把反斜杠去掉了。当shell再次扫描该行时,它替换了$4的值,并执行echo命令。
(2)对三重的支持
注意 这里的参数转义,第一遍处理后参数应该是:\$$aabbcc,第二次是:$alibaba,第三次就正好是:hello world。
3.用eval命令创建指向变量的“指针”:
x=100 ptrx=x eval echo \$$ptrx #指向ptrx 100 #打印100 eval $ptrx=50 # 将50存到ptrx指向的变量中。 echo $x 50 #打印50