1. eval command-line
其中command-line是在终端上键入的一条普通命令行。然而当在它前面放上eval时,其结果是shell在执行命令行之前扫描它两次。eval命令将会首先扫描命令行进行所有的替换,然后再执行命令。该命令使用于那些一次扫描无法实现其功能的变量。该命令对变量进行两次扫描。这些需要进行两次扫描的变量有时候被称为复杂变量。如:
pipe="|"
eval ls $pipe wc -l
如图:
shell第1次扫描命令行时,它替换出pipe的值|,接着eval使它再次扫描命令行,这时shell把|作为管道符号了。
如果变量中包含任何需要shell直接在命令行中看到的字符(不是替换的结果),就可以使用eval。命令行结束符(;| &),I/o重定向符(< >)和引号就属于对shell具有特殊意义的符号,必须直接出现在命令行中。
eval也可以用于回显简单变量,不一定是复杂变量。
如:
NAME=NO
eval echo $NAME等价于echo $NAME
2. eval echo \$$#取得最后一个参数
如:cat last
eval echo \$$#
./last one two three four
four
如图:
第一遍扫描后,shell把反斜杠去掉了。当shell再次扫描该行时,它替换了$4的值,并执行echo命令
3.以下示意如何用eval命令创建指向变量的“指针”:
x=100
ptrx=x
eval echo \$$ptrx指向ptrx,用这里的方法可以理解b中的例子
100 打印100
eval $ptrx=50将50存到ptrx指向的变量中。
echo $x
50 打印50
如图:
【与``和$()的区别】
在$(命令)中,命令部分的内容不会被父shell解析,而反引号的部分却会被解析,例如`\`,`$`,```
举个例子
> $(echo "first second"|awk "\$2==\"second\" {print $1}") 执行是成功的,执行结果为first
而
> a=`echo "first second"|awk "\$2==\"second\" {print $1}"`执行失败
> awk: line 1: syntax error at or near ==
原因是反引号内的内容会被父shell解释,所以\$2被父shell解释后变成了$2, 而$2在子shell里面是会被解析的,而默认的
$2代表子shell的第二个参数,显然为空,然后继续被awk解释,然后就报了上述的错误
其实还有另外一个错误,print $1,这里的$符号也是要被父shell解释的,则是不对的
最后改为
> `echo "first second"|awk "\\$2==\"second\" {print \\$2}`
1.反引号其本身就对\进行了转义,保留了其本身意思,如果我们想在反引号中起到\的特殊意义,我们必须使用2个\来进行表示。
2.所以我们可以简单的想象成反引号中: \\ = \
3.$()中则不需要考虑\的问题,与我们平常使用的一样:\ = \