|

楼主 |
发表于 2006-3-15 23:06:33
|
显示全部楼层
Post by wplxb
总算彻底想通了,呵呵
`command`格式中,“\”被shell处理为转义字符(如果后跟特殊含义字符$,\,`的话),因此`echo \$x`的结果相当于直接在命令行中输入echo $x并回车得到的结果,于是执行echo `echo \$x`得到结果为a(变量x的值)
而$(command)格式中,“\”被shell处理为普通字符,因此$(echo \$x)的结果相当于直接在命令行中输入echo \$x并回车得到的结果,于是执行echo $(echo \$x)得到结果为$x
我觉得命令替换可以理解为将要替换的命令传递给一个子shell执行,然后将得到的结果返回到被替换命令的位置继续执行父shell里的命令。`echo \$x`中,被传递给子shell的实际是echo $x;而$(echo \$x)中被传递给子shell的就是echo \$x
以下这段是引用了置顶贴home_king所写的bash命令行祥解
Post by home_king
1. 将命令行分成由固定元字符集分隔的记号:
SPACE, TAB, NEWLINE, ; , (, ), <, >, |, &
记号类型包括单词,关键字,I/O重定向符和分号。
2.检测每个命令的第一个记号,查看是否为不带引号或反斜线的关键字。如果是一个开放的关键字,如if和其他控制结构起始字符串,function,{或(,则命令实际上为一复合命令。shell在内部对复合命令进行处理,读取下一个命令,并重复这一过程。如果关键字不是复合命令起始字符串(如then等一个控制结构中间出现的关键字),则给出语法错误信号。
3.依据别名列表检查每个命令的第一个关键字。如果找到相应匹配,则替换其别名定义,并退回第一步;否则进入第4步。该策略允许递归别名,还允许定义关键字别名。如alias procedure=function
4.执行大括号扩展,例如a{b,c}变成ab ac
5.如果~位于单词开头,用$HOME替换~。使用usr的主目录替换~user。
6.对任何以符号$开头的表达式执行参数(变量)替换
7.对形式$(string)的表达式进行命令替换
这里是嵌套的命令行处理。
8.计算形式为$((string))的算术表达式
9.把行的参数,命令和算术替换部分再次分成单词,这次它使用$IFS中的字符做分割符而不是步骤1的元字符集。
10.对出现*, ?, [ / ]对执行路径名扩展,也称为通配符扩展
11. 按命令优先级表(跳过别名),进行命令查寻
12.设置完I/O重定向和其他操作后执行该命令。
所以小弟认为命令应该有如下的解释过程
- bash# x=a
- bash# echo $(echo $x)
- 这时的$x是先被当前的bash替换成其值a,如果是在执行子shell时替换的,那么$x应该时空值,所以命令行被解释为
- bash# echo $(echo a)
- 接着在进行命令替换
- bash# a
复制代码
根据上面的替换过程,对于有backslash的情况如下
- bash# echo $(echo \$x)
- 因为$x被前置了backslash,所以并没有被替换,接着的命令替换过程中,\$x->$x,所以命令就被解释成
- bash# echo $x
- 因为在命令替换以后只存在arithemetic expansion, word splitting, pattern match, command search and execution。所以这里的$x就被当作普通字符echo出来了
- bash# $x
复制代码
以上是小弟的更人想法,因为这个问题感觉很难说清楚,所以小弟就这样写下来,不知道这样的判断过程是否正确,希望各位给予指点:thank |
|