典型的编程语言中,&
符号一般都与逻辑”与“操作有关。在C语言中&
表示按位与操作,而&&
则表示逻辑与。在Bash中,&&
同样也是表示逻辑与,但是&
的作用则就不同了。
单个的与符号通常可以发现在一个bash命令的行尾:
./myscript.py &
其作用是令该命令转到后台执行。对于这样的命令,系统会创建一个sub-shell来运行这个命令。同事,在执行改行命令的shell环境中,这个命令会立刻返回0并且继续下面的shell命令的执行。除此之外,在执行这个命令之后,terminal上会输出创建的sub-shell的线程ID(PID)。
$ ./myscript.py &
[1] 1337
注意按照这种方法分支出去的sub-shell的stdout会仍然关联到其parent-shell,也就是说你在当前的terminal中仍然可以发现这个后台进程的stdout输出。
通过&
分支出去的sub-shell的PID被存储在一个特殊的变量$!
中,
$ echo $!
1337
同事,你也可以通过jobs
命令来检查sub-shell
的信息
$ jobs
[1]+ Running ./myscript.py &
对于sub-shell,你可以通过fg
命令将其拉回当前的terminal。
如果有多个命令需要放到后台运行,可以采用如下方式:
./script.py & ./script2.py & ./script3.py &
在这个例子中,三个脚本会同时开始运行,且拥有各自独立的sub-shell环境。在shell脚本中,这个方法常常被用来利用计算机的多核性能来加速执行。
注:
如果你想创建个完全和当前的shell独立的后台进程(而不是想上面提到的用&
创建的,和当前shell的stdout关联的方法),可以使用nohup
命令。
Update at 2018/08/21
有时候在shell命令中可能看到这样的写法:
some_command > /dev/null 2>&1
这里前面的> /dev/null
好理解,代表将stdout
导向到/dev/null
,而后者代表将stderr
重定向到stdout
,或者说stderr
的输出等同于stdout
。由于stdout
已经被重定向到/dev/null
,那么这意味着stderr
也被重定向到了这个位置。
1是系统默认代表
stdout
的值,2代表stderr
关于这点的扩展阅读: