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的几种方法(可关闭终端)
-
直接在终端用nohup执行
- nohup java HelloServer &
- &表示后台运行
- 如果不加&,需要直接关闭终端,否则前台无法执行其他命令(如果按下ctrl + C,则会直接终止,因为是发送了interrupt,将sleep中断了)
-
写一个shell脚本,run.sh
- 注意脚本中必须要加上 &
#!/bin/sh java HelloServer &
-
写一个shell脚本,run.sh
- 注意使用这种方式运行,可以不加 &
- 但是运行脚本后,也是前台运行,不能ctrl+c(原因同1),可以将当前终端关闭
#!/bin/sh trap '' SIGHUP java HelloServer
-
简单说明
- 上面三种方式关闭终端后,HelloServer都会在后台一直运行
- 可以看到2的方式其实是最简单的
- 使用nohup通常要将标准输出和标准错误的日志重定向到/dev/null,因为线上会有如logback的日志组件记录日志追加到文件
其他方式可以考虑setsid或者jvm通过参数忽略信号或者使用python的方式
原理分析
用nohup启动,原因比较简单,因为nohup的意思即忽略SIGHUP信号,如果不用nohup则系统对SIGHUP信号的默认处理是终止收到该信号的进程
-
第二个,先了解一下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信号),该信号会被发送到该进程组中的每一个进程。
-
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)
总结
- 推荐优先使用python的方式运行Java进程
- 其次使用shell执行'java &'