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
下面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 //,输出结果就为/了。