`
LeoCheng_gnq
  • 浏览: 8028 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

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

 
阅读更多

问题描述:当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,但删掉那些无法连接的会话 
 
其他资料:
  1. Linux 技巧:让进程在后台可靠运行的几种方法,https://www.ibm.com/developerworks/cn/linux/l-cn-nohup/
分享到:
评论

相关推荐

    终端关闭后程序在后台继续运行的方法

    终端关闭后程序在后台继续运行的方法

    Python脚本后台运行的几种方式

    现在脚本正常运行,通过ps能看到进程号,此时直接关闭ssh终端(不是用exit命令,是直接通过putty的关闭按钮执行的), 再次登录后发现进程已经退出了。 通过后台启动的方式该问题已经解决,这里总结下,也方便我以后...

    linux下使用ssh启动停止weblogicf服务.docx

    在Linux环境中,管理和操作WebLogic服务通常涉及到启动、停止以及监控等任务,这些操作可以通过SSH(Secure Shell)在远程服务器上进行。WebLogic是Oracle公司的一款企业级Java应用服务器,常用于部署和管理Java应用...

    Linux后台运行java的jar包.docx

    - **局限性**:一旦按下`CTRL + C`或者直接关闭SSH窗口,程序将会停止运行。 #### 方式二:简单的后台运行 **命令示例**: ``` java -jar shareniu.jar && ``` - **特点**:在命令结尾添加`&&`可以让程序在后台...

    Linux下运行后台任务的方法.docx

    screen 是一个虚拟终端软件,直接在 Linux 系统里面启动了另外一个后台程序接管(维持)了你的终端会话,当你直接连接的终端 SSH 断开时他仍然让程序认为你的 SSH 持续链接着,避免了进程接收到中断信号而退出。...

    SSH登陆LINUX服务器命令.doc

    - `-f`:后台运行SSH命令,常与`-n`和`-N`一起使用。 - `-i identity_file`:指定RSA身份验证的私钥文件路径,默认为`~/.ssh/id_rsa`。 - `-n`:将标准输入重定向到/dev/null,防止SSH读取stdin。 - `-t`:强制...

    Linux_SSH命令大全

    `nohup` 命令可以在后台运行程序,即使关闭终端窗口也不会影响程序执行。这里将 `wget` 命令放入后台运行,方便用户继续使用终端进行其他操作。 4. **配置 `wget` 防止意外中断**: ```bash wget -c ...

    Linux中实用但很小众的11个炫酷终端命令.doc

    Linux 终端命令大全 Linux 终端命令大全是 Linux 操作系统中最基本的交互方式,通过命令行可以实现各种操作和管理任务。今天,我们将分享 11 个炫酷的 Linux 终端命令大全,帮助您提升工作效率和解决常见问题。 1....

    Linux中jar包启动和jar包后台运行的实现方式

    &代表在后台运行。 特定:当前ssh窗口不被锁定,但是当窗口关闭时,程序中止运行。 继续改进,如何让窗口关闭时,程序仍然运行? 方式三 nohup java -jar shareniu.jar & nohup 意思是不挂断运行命令,当账户退出或...

    Linux 运行jar包的几种方式

    总的来说,Linux系统提供了多种方式来管理和控制jar包的运行,包括在后台运行、忽略输出、保持程序在SSH断开后继续运行等。理解这些命令的用法和组合对于日常的服务器管理、自动化部署和监控Java应用程序至关重要。

    linux 后台运行node服务指令方法

    通常,当我们在终端中直接运行`node server.js`时,如果关闭终端窗口或者断开SSH连接,Node.js服务也会随之停止。为了解决这个问题,我们可以使用一些特定的指令来让Node.js服务在后台持续运行。本文将详细介绍如何...

    linux--program.rar_linux程序

    7. **进程管理**:Linux提供丰富的工具管理进程,如`ps`查看进程,`kill`发送信号,`nohup`让程序在后台运行,`top`或`htop`实时监控系统资源占用。 8. **权限与所有权**:Linux的文件系统基于用户和组的概念,拥有...

    11个炫酷的Linux终端命令大全.docx

    、暂停并在后台运行命令、nohup 命令等。 一、命令行快捷键 命令行快捷键是提高工作效率的重要工具。以下是一些常用的快捷键: * CTRL + U: 剪切光标前的内容 * CTRL + K: 剪切光标至行末的内容 * CTRL + Y: 粘贴...

    jar畅游Linux后台

    目录jar通常方式jar后台运行方式方式一方式二查看后台运行任务查看某端口占用的线程的pidlinux 进程查看及杀死进程附录:各种信号及其用途 Linux系统运行jar包 jar通常方式 java -jar xxx.jar 特点:当前ssh窗口被...

    如何编写linux服务.zip

    在Linux系统中,服务是后台运行的程序,它们在用户登录之前启动,为系统提供持续性的功能。编写Linux服务通常涉及到创建脚本、配置启动机制以及管理服务的生命周期。以下是对"如何编写Linux服务"这一主题的详细说明...

    几个比较好用得Linux的命令

    本文将详细介绍几个非常实用的Linux命令,包括如何在后台运行程序、监控程序运行状态以及处理常见问题等。 #### 一、后台进程方式训练模型 ##### 1.1 登录服务器并执行命令 在进行模型训练之前,首先需要登录到...

    linux开启telnet服务

    Linux 提供服务是由运行在后台的守护程序(daemon)来执行的。守护进程的工作就是打开一个端口(port),等待(listen)进入的连接。在 C/S 模式中,如果客户提请了一个连接,守护进程就创建(fork)子进程来响应这...

Global site tag (gtag.js) - Google Analytics