美文网首页
关闭终端进程不退出的方法

关闭终端进程不退出的方法

作者: landon30 | 来源:发表于2018-08-27 21:32 被阅读0次

    Java的一个例子

    • sample代码很简单,main函数一个while(true),sleep
    • 我们的目标是要让这个sample在linux一直运行
    • 直接javac HelloServer.java进行编译
    public class HelloSever
    {
            public static void main(String...args) throws Exception
            {
                    while(true)
                    {
                            Thread.sleep(5000);
                            System.out.println("Hello,Server");
                    }
            }
    }
    

    在linux终端始终运行HelloServer的几种方法(可关闭终端)

    1. 直接在终端用nohup执行

      • nohup java HelloServer &
      • &表示后台运行
      • 如果不加&,需要直接关闭终端,否则前台无法执行其他命令(如果按下ctrl + C,则会直接终止,因为是发送了interrupt,将sleep中断了)
    2. 写一个shell脚本,run.sh

      • 注意脚本中必须要加上 &
      #!/bin/sh
      
      java HelloServer &
      
    3. 写一个shell脚本,run.sh

      • 注意使用这种方式运行,可以不加 &
      • 但是运行脚本后,也是前台运行,不能ctrl+c(原因同1),可以将当前终端关闭
      #!/bin/sh
      
      trap '' SIGHUP
      java HelloServer
      
    4. 简单说明

      • 上面三种方式关闭终端后,HelloServer都会在后台一直运行
      • 可以看到2的方式其实是最简单的
      • 使用nohup通常要将标准输出和标准错误的日志重定向到/dev/null,因为线上会有如logback的日志组件记录日志追加到文件
    5. 其他方式可以考虑setsid或者jvm通过参数忽略信号或者使用python的方式

    原理分析

    1. 用nohup启动,原因比较简单,因为nohup的意思即忽略SIGHUP信号,如果不用nohup则系统对SIGHUP信号的默认处理是终止收到该信号的进程

    2. 第二个,先了解一下sighup发生了什么

      • run.sh执行关闭,sh退出,java进程变为了孤儿进程(用ps -elf参数可以看到父进程变为了1,即孤儿进程)
      • java进程不是直接在终端启动的后台进程,所以关闭终端时不会收到sighup(即不属于终端的子进程)
      • 当session关闭发送sighup时,是会发送给所有前台进程的,而java进程是&运行,所以不会收到sighup
      • java进程是属于孤儿进程组,但是其没有处于stop,所以也不会收到sighup
      SIGHUP会在以下3种情况下被发送给相应的进程:
        1、终端关闭时,该信号被发送到session首进程以及作为job提交的进程(即用 & 符号提交的进程)
        2、session首进程退出时,该信号被发送到该session中的前台进程组中的每一个进程
        3、若父进程退出导致进程组成为孤儿进程组,且该进程组中有进程处于停止状态(收到SIGSTOP或SIGTSTP信号),该信号会被发送到该进程组中的每一个进程。
      
    3. trap本意是用于接收到指定信号后将要采取的动作

      • 当命令为空时,则表示指定的信号接收时,将被忽略
      • 则当关闭终端时,用trap忽略了sighup信号

    使用python

    • 可以使用python启动java,相比shell,pyhon更强大,直接看例子

      • 本例是引入了一个sh的库,先获取java命令,然后通过bake指定参数
      • 最后一步很重要,即指定标准输出重定向和设置后台运行
      • _bg参数即可实现忽略SIGHUP
      • 注意:python启动后结束,重定向的日志不再输出(所以线上环境完全不建议使用System.out)
      #!/usr/bin/env python
      #coding=utf8
      
      import sh
      import time
      
      java = sh.Command("java")
      run = java.bake("HelloServer")
      run(_out="py.out",_bg=True)
      
      time.sleep(10)
      

    总结

    1. 推荐优先使用python的方式运行Java进程
    2. 其次使用shell执行'java &'

    参考

    1. Linux kill & Java shutdownhook
    2. SIGHUP信号与控制终端
    3. Linux 技巧:让进程在后台可靠运行的几种方法
    4. Python sh库用法
    5. Java的一些日志实现细节

    相关文章

      网友评论

          本文标题:关闭终端进程不退出的方法

          本文链接:https://www.haomeiwen.com/subject/ngfxwftx.html