有些时候,我们需要在终端启动一个程序,并使之运行——但是如果关闭终端,那么这个程序也就随着关闭了。那么有没有什么方法在关闭终端后,让已经从这个终端启动的程序继续运行呢?
前置知识:
xterm,console,tty,pts,pty的区别
- shell是直接和内核进行通信的东西
- xterm是一个软件概念,可以通过这个程序连接到console从而控制主机,可以理解为cli形式的终端模拟器,而gnome-terminal,konsole就是gui形式的终端模拟器
- console是主机的控制台,是一个物理概念。
- tty、pty、pts都是终端,是硬件或者设备概念。
- tty是所有终端设备的总称
- pty是其中一类,伪终端,或者叫虚拟终
(2)“&” 命令行结尾法:
在Unix/Linux下如果想让程序独立终端运行,一般都是使用 & 在命令结尾来让程序自动运行。(命令后可以不追加空格)
打开gnome-terminal,执行如下命令:
delectate@delectate:~$ [1] 8510 delectate@delectate:~$
有几点需要注意:
- 已经启动的程序依然attach于当前pts,只有当前终端模拟器关闭(使用exit命令退出),进程自动被tty继承。
- delectate@delectate:~$ ps -e | grep totem //程序已被以totem & 形式启动,当前附在pts0上 8819 pts/0 00:00:00 totem delectate@delectate:~$ ps -e | grep totem 8819 ? 00:00:00 totem delectate@delectate:~$
- 具有debug输出的进程,需要按enter键进行中断当前debug输出。但是如果程序持续进行printf,你将无法输入任何命令。delectate@delectate:~$ [1] 8850 delectate@delectate:~$ VLC media player 1.0.6 Goldeneye [0x8b998b0] main libvlc: Running vlc with the default interface. Use 'cvlc' to use vlc without interface. //enter pressed delectate@delectate:~$ //show a clean terminal now ** (:8850): CRITICAL **: giop_thread_request_push: assertion `tdata != NULL' failed //仍然在输出数据…… //关闭程序 [1]+ Done vlc delectate@delectate:~$
- 你无法记录程序的debug输出结果。
- 只有当虚拟终端是 $ 或者 # 时候,才可以关闭此终端,否则可能导致已经启动的进程被关闭(按enter——如果程序持续输出信息而没有出现 $ 或 #)
(2).使用nohup命令:
nohup描述:Run COMMAND, ignoring hangup signals.(忽略任何中断/挂起信号,使命令继续执行)
nohup &
混用后,它会自动把你执行的命令输出结果记录到权限为-rw——-,名为nohup.out的文件中。
但是你仍然需要
delectate@delectate:~$ [1] 9045 delectate@delectate:~$ nohup: ignoring input and appending output to `nohup.out' //在这里按一下回车或以ctrl+c以 //show a clean terminal delectate@delectate:~$
与使用 “&” 性质相同,当前启动程序的终端如果没有被关闭,已经启动的程序附在pst上;如果终端被关闭,则自动附在tty。
如果当前目录的 nohup.out 文件不可写,输出重定向到 $HOME/nohup.out。默认状态下,nohup默认输出到nohup.out文件,你也可以利用重定向来指定输出文件:
nohup command {option} > myout.file 2>&1 &
只有当虚拟终端是 $ 或者 # 时候,才可以关闭此终端,否则可能导致已经启动的进程被关闭(按enter——如果程序持续输出信息而没有出现 $ 或 #)
(3)其他相关命令:
jobs:查看当前有多少在后台运行的命令
fg:将后台中的命令调至前台继续运行。如果后台中有多个命令,可以用 fg %jobnumber将选中的命令调出,%jobnumber是通过jobs命令查到的后台正在执行的命令的序号(不是pid)
bg:将一个在后台暂停的命令,变成继续执行。如果后台中有多个命令,可以用bg %jobnumber将选中的命令调出,%jobnumber是通过jobs命令查到的后台正在执行的命令的序号(不是pid)
(4)杀死进程
杀死已经启动的程序和普通方式一样:
pkill -9 name killall name kill pid …