bash命令行解释顺序

 

Shell从标准输入或脚本中读取的每行称为一个管道行,它包含一个或多个由0个或多个管道字符(|)分隔的命令。对每一个管道行,进行12个步骤的处理。
结合上面的插图,这里给出命令行的12个步骤。

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重定向和其他操作后执行该命令。


关于引用
1. 单引号跳过了前10个步骤,不能在单引号里放单引号
2. 双引号跳过了步骤1~5,步骤9~10,也就是说,只处理6~8个步骤。
也就是说,双引号忽略了管道字符,别名,~替换,通配符扩展,和通过分隔符分裂成单词。
双引号里的单引号没有作用,但双引号允许参数替换,命令替换和算术表达式求值。可以在双引号里包含双引号,方式是加上转义符"/",还必须转义$, `, /。

摘自man bash 的一段话来解释双引号:
    Enclosing characters in double quotes preserves the literal value of all characters within the  quotes,  with  the exception  of  $,  `,  /,  and, when history expansion is enabled, !.  The characters $ and ` retain their special
meaning within double quotes.  The backslash retains its special meaning only when followed by one of the  following  characters: $, `, ", /, or .   A double quote may be quoted within double quotes by preceding it with a backslash.  If enabled, history expansion will be performed unless an !  appearing in double quotes  is  escaped using a backslash.  The backslash preceding the !  is not removed.

下面man bash的一段话来解释命令替换$()和``
    When  the old-style backquote form of substitution is used, backslash retains its literal meaning except when followed by $, `, or /.  The first backquote not preceded by a backslash terminates the command  substitution.   When using the $(command) form, all characters between the parentheses make up the command; none are treated specially.



要思考的问题:
1.echo `echo ///z` 的输出 和 echo `echo z` 的输出。

2.
在bash中:
$echo "//"
输出:/

$A='//'
$echo "$A"
输出://
解释原因。

原因:这与bash命令行处理的顺序有关。bash中对引用(单双引号和/)的处理在对参数扩展(展开变量)之前,所以将$A的值代入命令行之后bash就不再解释转义或称作逃逸字符。有时为了让shell再次进行命令行的一系列处理,需要使用eval。

其实有两种方法让输出结果为/:
方法一、echo -e "$A"
方法二、eval echo "$A"
其中方法一是通过改变 echo命令的执行方式达到结果,
方法二是通过改变shell处理来达到结果。

3.echo `echo //` 与 echo $(echo //)的输出分别是什么?解释原因。
原因: ``里面的/是一个特殊字符,可以用它来引用特殊的字符(当然包括它自身/),而$()里面的/只是普通字符。
echo `echo //`命令,里层的echo //得到的结果/,于是外层命令为echo /,输出结果就为空了。
echo $(echo //)命令,里层的/不再作为特殊字符,其输出就是//,于是外层命令为echo //,输出结果就为/了。

你可能感兴趣的:(转载备份)