shell:解释执行用户输入命令,逐行解释,也可以批处理很多指令,可将这些指令全部编写到一个脚本文件中,让shell一次执行多条命令,常用的shell(bash,它承担shell这个角色,为shell脚本解释器,为一个可执行的二进制文件)

shell的执行过程:

用户在命令行输入命令后,一般情况下Shell会fork子进程并调用exec进行程序替换,让它的孙进程执行该命令,但是Shell的内建命令例外,执行内建命令相当于调用Shell进程中的一个函数,并不创建新的进程。如cd、export、umask、exit等命令即是内建命令

具体步骤:

1.交互Shell(bash)fork/exec一个子Shell(sh)用于执行脚本,父进程bash等待子进程sh终止。

2.sh读取脚本中的cd ..命令,调用相应的函数执行内建命令,改变当前工作目录为上一级目录。

3.sh读取脚本中的ls命令,fork/exec这个程序,列出当前工作目录下的文件,sh等待ls终止。

4.ls终止后,sh继续执行,读到脚本文件末尾,sh终止。

5.sh终止后,bash继续执行,打印提示符等待用户输入。

eval:是在终端上键入的一条普通命令行。然而当在它前面放上eval时,其结果是shell在执行命令行之前扫描它两次

用法:eval [参数]    eval可读取一连串的参数,然后再依参数本身的特性来执行。

参数:参数行之前扫描它两次,再次运算求出参数的内容。

1.用于回显简单变量:

2.两次扫描,用于复杂变量显示,新建一个文本文件,显示该文本文件内容

3.可获取最后一个参数

4.用eval创建指针变量,也可以改变该指针所指空间的内容

eval:不能获得函数处理结果 ,所有命令,函数的处理结果只能通过 ``来获得。eval 嵌套无意义 ,在其他语言中可以通过 eval(eval("code")),来执行(执行动态生成的code的返回),而由于shell 中 eval 将后面的eval命令简单当作命令字符串执行,失去了嵌套作用,嵌套被命令替换取代。

` `与$()的区别与联系

1.在有些情况下它们的用法相同,但它们之间也有区别

如上例,显示当前时间时它们所获得的值是一样的,但获得当前目录所在主目录时,``将$的特殊意义转换出来了,而使用$()时并未解释出来,被\转义成了一个普通字符

2.若给上例再加上一个/

$()被转义成了一个字符串,并内解释出内容

3.在test.sh脚本中编写如下内容

如上,用反引号时,四个反斜杠被解释成了一个,而用$()时被解释成了两个

4.使用多个反斜杠时

由上可得:反引号本身就对\进行了转义,保留了其本身意思,如果想在反引号中起到\的特殊意义,必须使用两个\表示,$()使用时与平时一样 \=\

[测试命令与[[]]的区别与联系

1.[是一个可执行程序

2.单括号TEST命令要对变量进行单词分离,当变量值包含空白符时,要用引号将变量括起来;而双括号的TEST命令不会对变量进行单词分离

当要测试时,bash回应参数太多了,是因为单引号TEST命令对变量进行单词分离了,结果也就变成了

[ hello world = "hello world" ],多了那个字符串hello,成了字符串world和字符串hello world之间的比较了。因此如果在单括号的TEST命令的变量含有空格,但它还得和字符串比较,那就必须给变量加个双引号了,这时它就不会发生错误了。而在下面的双括号的TEST命令不能对变量单词进行分离

3.单括号的TEST命令不对SHELL元字符进行扩展的,而双括号TEST命令则会对SHELL元字符进行扩展的了。

4.在双括号的TEST命令中,如果一个字符串(不管含不含有空格)仅仅是在表达式作为一个普通字符串,而不是一个模式的一部分,则它也必须用引号括起来。如果一个字符串值(右边那个的了)不加双引号,那这个字符串就是模式来的,如果它里面有含有SHELL元字符,BASH会对它进行扩展。如果字符串加了双引号,那它就是一个很普通的字符串的,即便字符串里面里面含有特殊字符,也就是当普通内容来处理。

5.[ express1 –a express2 ] 这是放在单括号的TEST命令的,因为单的不支持元字符扩展,因此就只能叫做表达式了,它们可以组合构成逻辑测试的