当前位置:操作系统 > Unix/Linux >>

详解Bash命令行处理

很多兄弟写脚本或命令时出现错误的主要原因,是因为不了解bash的命令行处理。我在这里总结了一下,大家可以参考一下。其中也涉及到双引号,单引号以及eval的技巧,我会一一讲述。

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

  

  一、bash命令处理的12个步骤;

  
                               +-------------+           单引号

  |------------------------->| |--------------------------|

  | ----------------------->| 1.分隔成记号|---- ---------------| |

  | | ------------------->| | 双引号 | |

  | | | +-------------+ | |

  | | | || | |

  | | |读取下一个命令 \/ | |

  | | | +-------------------------------------------+ | |

  | | | | 2. | | |

  | | ------| 检验第一个记号 | | |

  | | |开放的关键字 其他关键字 | | |

  | | | 非关键字 | | |

  | | +-------------------------------------------+ | |

  | | || | |

  | | \/ | |

  | | +-----------------------------+ | |

  | | 扩展别名 | 3. 检验第一个记号 | | |

  | |------------| 别名 | | |

  | | 不是别名 | | |

  | +-----------------------------+ | |

  | || | |

  | \/ | |

  | +--------------+ | |

  | | 4.大括号扩展 | | |

  | +--------------+ | |

  | || | |

  | \/ | |

  | +--------------+ | |

  | | 5.~符号扩展 | | |

  | +--------------+ | |

  | || | |

  | \/ | |

  | +--------------+ 双引号 | |

  | | 6.参数扩展 |<-----------------| |

  | +--------------+ |

  | || |

  | \/ |

  | +------------------------------+ |

  | | 7.命令替换(嵌套命令行处理) | |

  | +------------------------------+ |

  | || |

  | \/ |

  | +--------------+ 双引号 |

  | | 8.算术扩展 |------------------| |

  | +--------------+ | |

  | || | |

  | \/ | |

  | +--------------+ | |

  | | 9.单词分割 | | |

  | +--------------+ | |

  | || | |

  | \/ | |

  | +--------------+ | |

  | | 10.路径名扩展| | |

  | +--------------+ | |

  | || | |

  | \/ | |

  | +----------------------------------------+ | |

  | | 11.命令查寻:函数,内置命令,可执行文件|<---|-----|

  | +----------------------------------------+

  | ||

  | \/

  |将参数带入下一个命令 +-------------+

  |----------eval--------------| 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个步骤。

  也就是说,双引号忽略了管道字符,别名,~替换,通配符扩展,和通过分隔符分裂成单词。

  双引号里的单引号没
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,