解决Linux关闭终端(关闭SSH等)后运行的程序或者服务自动中止

问题描述:当SSH远程链接到服务器上,而后运行一个服务 ./catalina.sh start,而后把终端开闭(切断SSH链接)以后,发现该服务中断,致使网页没法访问。

解决方法:使用nohup命令让程序在关闭窗口(切换SSH链接)的时候程序还能继续在后台运行。php

Unix/Linux下通常好比想让某个程序在后台运行,不少都是使用& 在程序结尾来让程序自动运行。好比咱们要运行mysql在后台:mysql

/usr/local/mysql/bin/mysqld_safe --user=mysql &

可是加入咱们不少程序并不象mysqld同样作成守护进程,可能咱们的程序只是普通程序而已,通常这种程序使用& 结尾,可是若是终端关闭,那么程序也会被关闭。可是为了可以后台运行,那么咱们就可使用nohup这个命令,好比咱们有个test.php须要在后台运行,而且但愿在后台可以按期运行,那么就使用nohup:sql

nohup /root/test.php &

  提示:shell

 

 [~]$ appending output to nohup.out

  嗯,证实运行成功,同时把程序运行的输出信息放到当前目录的nohup.out 文件中去。segmentfault

nohup命令说明:

  用途:不挂断地运行命令。bash

  语法:nohup Command [ Arg ... ] [ & ]服务器

  描述:nohup 命令运行由 Command 参数和任何相关的 Arg 参数指定的命令,忽略全部挂断(SIGHUP)信号。在注销后使用 nohup 命令运行后台中的程序。要运行后台中的 nohup 命令,添加 & ( 表示“and”的符号)到命令的尾部。网络

  不管是否将 nohup 命令的输出重定向到终端,输出都将附加到当前目录的 nohup.out 文件中。若是当前目录的 nohup.out 文件不可写,输出重定向到 $HOME/nohup.out 文件中。若是没有文件能建立或打开以用于追加,那么 Command 参数指定的命令不可调用。若是标准错误是一个终端,那么把指定的命令写给标准错误的全部输出做为标准输出重定向到相同的文件描述符。session

  

退出状态:该命令返回下列出口值:

  126 能够查找但不能调用 Command 参数指定的命令。app

  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/... )

结论:所以当网络断开或终端窗口关闭后,也就是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,但删掉那些没法链接的会话

 

 

2 条评论

WickedDogg · 2017年09月21日

简简单单打个exit的事。。。nohup是后台运行,不阻塞你在命令行接着作其它事

 赞 回复

恩,其实这个nohup主要仍是用来在操控远程服务器端实现运行某个命令的时候不会由于你关掉本地的命令行退出终端的时候把那个命令给杀掉,例如我想在服务器端运行一个php文件,若是我关闭了ssh,那么这个运行php文件的程序就会挂掉

— zero风来 做者 · 2017年09月21日

相关文章
相关标签/搜索