问题描述:当SSH远程连接到做事器上,然后运行一个做事 ./catalina.sh start,然后把终端开闭(割断SSH连接)之后,创造该做事中断,导致网页无法访问。
办理方法:利用nohup命令让程序在关闭窗口(切换SSH连接)的时候程序还能连续在后台运行。
Unix/Linux下一样平常比如想让某个程序在后台运行,很多都是利用& 在程序结尾来让程序自动运行。比如我们要运行mysql在后台:

/usr/local/mysql/bin/mysqld_safe --user=mysql &
但是加入我们很多程序并不象mysqld一样做成守护进程,可能我们的程序只是普通程序而已,一样平常这种程序利用& 结尾,但是如果终端关闭,那么程序也会被关闭。但是为了能够后台运行,那么我们就可以利用nohup这个命令,比如我们有个test.php须要在后台运行,并且希望在后台能够定期运行,那么就利用nohup:
nohup /root/test.php &
提示:
[~]$ appending output to nohup.out
嗯,证明运行成功,同时把程序运行的输出信息放到当前目录的nohup.out 文件中去。
nohup命令解释:
用场:不挂断地运行命令。
语法:nohup Command [ Arg ... ] [ & ]
描述:nohup 命令运行由 Command 参数和任何干系的 Arg 参数指定的命令,忽略所有挂断(SIGHUP)旗子暗记。在注销后利用 nohup 命令运行后台中的程序。要运行后台中的 nohup 命令,添加 & ( 表示“and”的符号)到命令的尾部。
无论是否将 nohup 命令的输出重定向到终端,输出都将附加到当前目录的 nohup.out 文件中。如果当前目录的 nohup.out 文件不可写,输出重定向到 $HOME/nohup.out 文件中。如果没有文件能创建或打开以用于追加,那么 Command 参数指定的命令不可调用。如果标准缺点是一个终端,那么把指定的命令写给标准缺点的所有输出作为标准输出重定向到相同的文件描述符。
退出状态:该命令返回下列出口值:
126 可以查找但不能调用 Command 参数指定的命令。
127 nohup 命令发生缺点或不能查找由 Command 参数指定的命令。
否则,nohup 命令的退出状态是 Command 参数指定命令的退出状态。
nohup命令及其输出文件
nohup命令:如果你正在运行一个进程,而且你以为在退出帐户时该进程还不会结束,那么可以利用nohup命令。该命令可以在你退出帐户/关闭终端之后连续运行相应的进程。nohup便是不挂起的意思( n ohang up)。
该命令的一样平常形式为:nohup command &
利用nohup命令提交作业
如果利用nohup命令提交作业,那么在缺省情形下该作业的所有输出都被重定向到一个名为nohup.out的文件中,除非其余指定了输出文件:(也便是说自定义输出的文件名)
nohup command > myout.file 2>&1 &
在上面的例子中,输出被重定向到myout.file文件中。
利用 jobs 查看任务。
利用 fg %n 关闭。
其余有两个常用的ftp工具ncftpget和ncftpput,可以实现后台的ftp上传和下载,这样我就可以利用这些命令在后台上传和下载文件了。
思考:问题1为什么ssh一关闭,程序就不再运行了?
首恶:SIGHUP 旗子暗记
让我们来看看为什么关掉窗口/断开连接会使得正在运行的程序去世掉。
在Linux/Unix中,有这样几个观点:
进程组(process group):一个或多个进程的凑集,每一个进程组有唯一一个进程组ID,即进程组长进程的ID。
会话期(session):一个或多个进程组的凑集,有唯一一个会话期首进程(session leader)。会话期ID为首进程的ID。
会话期可以有一个单独的掌握终端(controlling terminal)。与掌握终端连接的会话期首进程叫做掌握进程(controlling process)。当前与终端交互的进程称为前台进程组。别的进程组称为后台进程组。
根据POSIX.1定义:
挂断旗子暗记(SIGHUP)默认的动作是终止程序。
当终端接口检测到网络连接断开,将挂断旗子暗记发送给掌握进程(会话期首进程)。
如果会话期首进程终止,则该旗子暗记发送到该会话期前台进程组。
一个进程退出导致一个孤儿进程组中产生时,如果任意一个孤儿进程组进程处于STOP状态,发送SIGHUP和SIGCONT旗子暗记到该进程组中所有进程。(关于孤儿进程参照:http://blog.csdn.net/hmsiwtv/article/details/7901711 )
结论:因此当网络断开或终端窗口关闭后,也便是SSH断开往后,掌握进程收到SIGHUP旗子暗记退出,会导致该会话期内其他进程退出。
简而言之:便是ssh 打开往后,bash等都是他的子程序,一旦ssh关闭,系统将所有干系进程杀掉!
!
导致一旦ssh关闭,实行中的任务就取消了
例子:
我们来看一个例子。打开两个SSH终端窗口,在个中一个运行top命令。
[root@tivf09 root]# top
在另一个终端窗口,找到top的进程ID为5180,其父进程ID为5128,即登录shell。
[root@tivf09 root]# ps -ef|grep top
root 5180 5128 0 01:03 pts/0 00:00:02 top
root 5857 3672 0 01:12 pts/2 00:00:00 grep top
利用pstree命令可以更清楚地看到这个关系:
[root@tivf09 root]# pstree -H 5180|grep top
|-sshd-+-sshd---bash---top
利用ps-xj命令可以看到,登录shell(PID 5128)和top在同一个会话期,shell为会话期首进程,所在进程组PGID为5128,top所在进程组PGID为5180,为前台进程组。
[root@tivf09 root]# ps -xj|grep 5128
5126 5128 5128 5128 pts/0 5180 S 0 0:00 -bash
5128 5180 5180 5128 pts/0 5180 S 0 0:50 top
3672 18095 18094 3672 pts/2 18094 S 0 0:00 grep 5128
关闭第一个SSH窗口,在另一个窗口中可以看到top也被杀掉了。
[root@tivf09 root]# ps -ef|grep 5128
root 18699 3672 0 04:35 pts/2 00:00:00 grep 5128
问题2 为什么守护程序就算ssh 打开的,就算关闭ssh也不会影响其运行?
由于他们的程序分外,比如httpd –k start运行这个往后,他不属于sshd这个进程组 而是单独的进程组,以是就算关闭了ssh,和他也没有任何关系!
[root@CentOS5-4 ~]# pstree |grep http
|-httpd
[root@CentOS5-4 ~]# pstree |grep top
|-sshd-+-sshd---bash---top
结论:守护进程的启动命令本身便是分外的,和一样平常命令不同的,比如mysqld_safe 这样的命令 一旦利用了 便是守护进程运行。以是想把一样平常程序改造为守护程序是不可能,
问题3 利用后台运行命令& 能否将程序摆脱ssh进程组掌握呢 也便是ssh关闭,后台程序连续运行?
我们做一个试验: find / -name ‘http’&
利用ctrl+d 注销往后 再进入系统 会不会瞥见这个命令再运行?
答案是 :命令被中止了!!
由于他依然属于这个ssh进程组 就算加了&也无法摆脱!
!
[root@CentOS5-4 ~]# pstree |grep find
|-sshd-+-sshd---bash---find
结论便是:只假如ssh 打开实行的一样平常命令,不是守护程序,无论加不加&,一旦关闭ssh,系统就会用SIGHUP终止
问题4 nohup能办理的问题
但是为了能够再注销往后 依然能后台运行,那么我们就可以利用nohup这个命令,我们现在开始查找find / -name ‘http’&
,并且希望在后台运行,
那么就利用nohup:nohup find / -name \"大众httpd\公众
此时默认地程序运行的输出信息放到当前文件夹的nohup.out 文件中去
加不加&并不会影响这个命令 只是让程序 前台或者后台运行而已
延伸:Linux命令nohup+screen命令
如果想在关闭ssh连接后刚才启动的程序连续运行怎么办,可以利用nohup。但是如果哀求第二天来的时候,一开ssh,还能查看到昨天运行的程序的状态,然后连续事情,这时nohup是弗成了,须要利用screen来达到这个目的。
虽然nohup很随意马虎利用,但还是比较“简陋”的,对付大略的命令能够搪塞过来,对付繁芜的须要人机交互的任务就麻烦了。
实在我们可以利用一个更为强大的实用程序screen。盛行的Linux发行版(例如Red Hat Enterprise Linux 4)常日会自带screen实用程序,如果没有的话,可以从GNU screen的官方网站下载。
1)利用
实行screen , 按任意键进入子界面;
我用ping命令开始实行,如果放工了,但是想关闭ssh往后ping连续运行,那么按ctrl+a 再按d 这样停息了子界面,会显示[detached]的字样,这时候 我回到了父界面;
用screen –ls查看目前子界面的状态screen -ls
There is a screen on: 22292.pts-3.free (Detached)
1 Socket in /tmp/screens/S-root,这里的22292实在是子界面的pid号;
如果回到子界面 用screen –r 22292,一下子弹到了ping 的子界面;
2)更多帮助
可以通过C-a(ctrl+a)?来查看所有的键绑定,常用的键绑定有:
C-a ?
显示所有键绑定信息
C-a w
显示所有窗口列表
C-a C-a
切换到之前显示的窗口
C-a c
创建一个新的运行shell的窗口并切换到该窗口
C-a n
切换到下一个窗口
C-a p
切换到前一个窗口(与C-a n相对)
C-a 0..9
切换到窗口0..9
C-a a
发送C-a到当前窗口
C-a d
暂时断开screen会话
C-a k
杀掉当前窗口
C-a [
进入拷贝/回滚模式
其他常用选项:
-c file
利用配置文件file,而不该用默认的$HOME/.screenrc
-d|-D [pid.tty.host]
不开启新的screen会话,而是断开其他正在运行的screen会话
-h num
指定历史回滚缓冲区大小为num行
-list|-ls
列涌现有screen会话,格式为pid.tty.host
-d -m
启动一个开始就处于断开模式的会话
-r sessionowner/ [pid.tty.host]
重新连接一个断开的会话。多用户模式下连接到其他用户screen会话须要指定sessionowner,须要setuid-root权限
-S sessionname
创建screen会话时为会话指定一个名字
-v
显示screen版本信息
-wipe [match]
同-list,但删掉那些无法连接的会话
其他资料:
Linux 技巧:让进程在后台可靠运行的几种方法,https://www.ibm.com/developerworks/cn/linux/l-cn-nohup/