- 论坛徽章:
- 0
|
+-------------+ 单引号
|------------------------->| |--------------------------|
| ----------------------->| 1.分隔成记号|---- ---------------| |
| | ------------------->| | 双引号 | |
| | | +-------------+ | |
| | | || | |
| | |读取下一个命令 \/ | |
| | | +-------------------------------------------+ | |
| | | | 2. | | |
| | ------| 检验第一个记号 | | |
| | |开放的关键字 其他关键字 | | |
| | | 非关键字 | | |
| | +-------------------------------------------+ | |
| | || | |
| | \/ | |
| | +-----------------------------+ | |
| | 扩展别名 | 3. 检验第一个记号 | | |
| |------------| 别名 | | |
| | 不是别名 | | |
| +-----------------------------+ | |
| || | |
| \/ | |
| +--------------+ | |
| | 4.大括号扩展 | | |
| +--------------+ | |
| || | |
| \/ | |
| +--------------+ | |
| | 5.~符号扩展 | | |
| +--------------+ | |
| || | |
| \/ | |
| +--------------+ 双引号 | |
| | 6.参数扩展 |
Shell从标准输入或脚本中读取的每行称为一个管道行,它包含一个或多个由0个或多个管道字符(|)分隔的命令。对每一个管道行,进行12个步骤的处理。
结合上面的插图,这里给出命令行的12个步骤。
1. [color="blue"]将命令行分成由固定元字符集分隔的记号:
[color="red"]SPACE, TAB, NEWLINE, ; , (, ), , |, &
记号类型包括单词,关键字,I/O重定向符和分号。
2.[color="blue"]检测每个命令的第一个记号,查看是否为不带引号或反斜线的关键字。如果是一个开放
的关键字,如if和其他控制结构起始字符串,function,{或(,则命令实际上为一复合命令。shell在内部对复合命令进行处理,读取下一个命
令,并重复这一过程。如果关键字不是复合命令起始字符串(如then等一个控制结构中间出现的关键字),则给出语法错误信号。
3.[color="blue"]依据别名列表检查每个命令的第一个关键字。如果找到相应匹配,则替换其别名定义,并退回第一步;否则进入第4步。该策略允许递归别名,还允许定义关键字别名。如alias procedure=function
4.[color="blue"]执行大括号扩展,例如a{b,c}变成ab ac
5.[color="blue"]如果~位于单词开头,用$HOME替换~。使用usr的主目录替换~user。
6.[color="blue"]对任何以符号$开头的表达式执行参数(变量)替换
7.[color="blue"]对形式$(string)的表达式进行命令替换
[color="red"]这里是嵌套的命令行处理。
8.[color="blue"]计算形式为$((string))的算术表达式
9.[color="blue"]把行的参数,命令和算术替换部分再次分成单词,这次它使用$IFS中的字符做分割符而不是步骤1的元字符集。
10.[color="blue"]对出现*, ?, [ / ]对执行路径名扩展,也称为通配符扩展
11. [color="blue"]按命令优先级表(跳过别名),进行命令查寻
12.[color="blue"]设置完I/O重定向和其他操作后执行该命令。
关于引用
1. 单引号跳过了前10个步骤,不能在单引号里放单引号
2. 双引号跳过了步骤1~5,步骤9~10,也就是说,只处理6~8个步骤。
也就是说,双引号忽略了管道字符,别名,~替换,通配符扩展,和通过分隔符分裂成单词。
双引号里的单引号没有作用,但双引号允许参数替换,命令替换和算术表达式求值。可以在双引号里包含双引号,方式是加上转义符"\",还必须转义$, `, \。
[color="blue"]摘自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.
[color="blue"]
要思考的问题:
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 \\,输出结果就为\了。
[color="blue"]
本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u2/63316/showart_2152062.html |
|