🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
## 子Shell 子Shell或者叫子进程,是从父子进程的概念出发的,UNIX操作系统的进程从init进程开始其他均有其对应的子进程,就算是由于父进程先行结束导致的孤儿进程,也会被init领养使其父进程ID为1。在Bash中 通常会通过以下几种形式产生子进程: * &,提交后台作业 * \|,管道 * \(\),操作符 * 外部命令 需要注意的是,在写Bash脚本过程中推荐_\*\*_使用Bash内建\(builtin\)命令,因为外部命令会forks一个子进程效率并不是很高,我们来看以下测试案例: ``` [root@blog.puppeter.com_centos ~]# time for i in `seq 1 1000`;do result=$(expr length $test);done real 0m0.764s user 0m0.437s sys 0m0.323s [root@blog.puppeter.com_centos ~]# time for i in `seq 1 1000`;do result={#test} ;done real 0m0.005s user 0m0.005s sys 0m0.000s ``` 案例主要是测试test字符串的长度。第一个案例调用了外部命令,第二个命令是调用内部命令可以看到最终的执行结果性能相差100倍之多,所以推荐在编写脚本过程中尽量使用内建命令。 ### 子Shell案例 #### 案例1 使用\(\)操作符来创建子进程。 ``` #!/bin/bash echo "Father Shell is: $BASH_SUBSHELL" # 打印父Shell的层次,为0 outervar=OUTER # 父Shell的变量outervar ( # 利用圆括号结构创建子Shell echo "SubShell is: $BASH_SUBSHELL" # 子Shell的层次为1 ( echo "GrandSubShell is: $BASH_SUBSHELL" # 孙Shell的层次为2 ) innervar=INNER # 子Shell的变量 echo "innervar=$innervar" echo "outervar=$outervar" # outervar继承了符Shell所赋给它的值 ) # 回到父shell echo "Father Shell is: $BASH_SUBSHELL" if [ -z "$innervar" ] # 子Shell中定义变量为空,则说明并没有获取子Shell中变量 then echo "The \$innervar is not defined in main body." else echo "The \$innervar is defined in main body." fi ``` #### 案例2 通过管道\| ,& 和()方式组成命令组并实现并发执行的效果,譬如以下这个案例: ``` #!/bin/bash (grep -r "root" /etc/* | sort > part1) & # 与root关键字匹配的行,排序后输出到某文件 (grep -r "root" /usr/local/* | sort > part2) & (grep -r "root" /lib/* | sort > part3) & wait # 等待后台执行的作业全部完成 cat part1 part2 part3 | sort > parttotal echo "Run time of this script is:$SECONDS" # 输出该脚本执行时间 ``` #### 案例3 有的同学也会关心\(\)和{}操作符的区别,其实他们都是对命令组执行: 相同点: * \(\)和{}都是把一串的命令放在括号里面,并且命令之间用;号隔开执行 不同点: * \(\)只是对一串命令重新开一个子Shell进行执行,{}对一串命令在当前Shell执行 * \(\)最后一个命令可以不用分号,{}最后一个命令要用分号 > 注意: 在使用{}时,{}与命令之间必须使用一个空格 看以下案例。 ``` [root@blog.puppeter.com_centos ~]# A=1;echo $A;{ A=2; };echo $A 1 2 [root@blog.puppeter.com_centos ~]# A=1;echo $A;(A=2);echo $A 1 1 ```