在Linux上将Java应用程序作为服务运行


128

我已经编写了一个在标准虚拟主机Linux解决方案上运行的Java服务器应用程序。该应用程序始终在运行,以监听套接字连接并为其创建新的处理程序。它是客户端服务器应用程序的服务器端实现。

我启动它的方法是将其包含在服务器的启动rc.local脚本中。但是,一旦启动,我不知道如何访问它以停止它以及是否要安装更新,因此我必须重新启动服务器才能重新启动应用程序。

在Windows PC上,对于这种类型的应用程序,我可以创建Windows服务,然后可以根据需要停止和启动它。在Linux机器上是否有类似的东西,所以如果我启动此应用程序,则可以停止它并重新启动它,而无需完全重新启动服务器。

我的应用程序称为WebServer.exe。它是通过将其包含在我的rc.local中的方式在服务器启动时启动的:

java -jar /var/www/vhosts/myweb.com/phpserv/WebServer.jar &

我在Linux上有点菜鸟,因此任何示例都将不胜感激。但是我确实有SSH,并且对安装该更新程序的盒子具有完全的FTP访问权限以及对Plesk面板的访问权限。

Answers:


239

我在这里写了另一个简单的包装:

#!/bin/sh
SERVICE_NAME=MyService
PATH_TO_JAR=/usr/local/MyProject/MyJar.jar
PID_PATH_NAME=/tmp/MyService-pid
case $1 in
    start)
        echo "Starting $SERVICE_NAME ..."
        if [ ! -f $PID_PATH_NAME ]; then
            nohup java -jar $PATH_TO_JAR /tmp 2>> /dev/null >> /dev/null &
            echo $! > $PID_PATH_NAME
            echo "$SERVICE_NAME started ..."
        else
            echo "$SERVICE_NAME is already running ..."
        fi
    ;;
    stop)
        if [ -f $PID_PATH_NAME ]; then
            PID=$(cat $PID_PATH_NAME);
            echo "$SERVICE_NAME stoping ..."
            kill $PID;
            echo "$SERVICE_NAME stopped ..."
            rm $PID_PATH_NAME
        else
            echo "$SERVICE_NAME is not running ..."
        fi
    ;;
    restart)
        if [ -f $PID_PATH_NAME ]; then
            PID=$(cat $PID_PATH_NAME);
            echo "$SERVICE_NAME stopping ...";
            kill $PID;
            echo "$SERVICE_NAME stopped ...";
            rm $PID_PATH_NAME
            echo "$SERVICE_NAME starting ..."
            nohup java -jar $PATH_TO_JAR /tmp 2>> /dev/null >> /dev/null &
            echo $! > $PID_PATH_NAME
            echo "$SERVICE_NAME started ..."
        else
            echo "$SERVICE_NAME is not running ..."
        fi
    ;;
esac 

您可以按照一个完整的教程的init.d 这里和systemd(Ubuntu的16+)这里

如果您需要输出日志,请更换 2

nohup java -jar $PATH_TO_JAR /tmp 2>> /dev/null >> /dev/null &

线

nohup java -jar $PATH_TO_JAR >> myService.out 2>&1&

@PbxMan对此表示感谢。我可能会去看看我们如何发展。干杯。
dreza 2014年

2
但是我该如何运行该文件?我必须放在哪里?
杰克·丹尼尔

3
在基于debian的发行版上的@JackDaniel,例如debian本身和ubuntu,您可以将该脚本添加到/etc/init.d中。然后,您可以像这样调用它:/etc/init.d/MyService start。您可以通过运行update-rc.d MyService defaults使其自动启动。
安德烈

1
@ThorbjørnRavnAndersen这将取决于您的Java程序。如果您无法杀死Java程序,请查看stackoverflow.com/questions/2541597/…。我将删除MyService-pid而不是kill,并在Java部分中检查它是否存在的守护进程线程。
PbxMan

1
罐子的输出文件在哪里?我该如何配置它的名称?
M. Schena

48

一个简单的解决方案是创建一个脚本start.sh,该脚本通过nohup运行Java,然后将PID存储到文件中:

nohup java -jar myapplication.jar > log.txt 2> errors.txt < /dev/null &
PID=$!
echo $PID > pid.txt

然后,您的停止脚本stop.sh将从文件中读取PID并终止该应用程序:

PID=$(cat pid.txt)
kill $PID

当然,我遗漏了一些细节,例如检查该进程是否存在以及是否已删除pid.txt


2
问题:kill $ PID命令不会导致进程不完成就被杀死吗?我正在编写一个与数据库连接的服务器程序,并且希望所有当前正在运行的线程在程序退出之前完成,以确保该程序不会在写入数据库或其他操作时死亡。
Scuba Steve

2
@ scuba-steve之类的。kill将发送TERM信号,该信号将调用任何适当的关闭挂钩,因此请使用它们优雅地结束您的进程。如果进程收到终止信号(即终止-9),它们将不会执行。如果它们花费的时间太长,操作系统可能会中断您的关闭挂钩,因此请保持简洁
rjohnston 2014年

34

Linux服务初始化脚本存储在中/etc/init.d。您可以复制和自定义/etc/init.d/skeleton文件,然后调用

service [yourservice] start|stop|restart

参见http://www.ralfebert.de/blog/java/debian_daemon/。它适用于Debian(因此也适用于Ubuntu),但适合更多发行版。


看起来很有希望。我将对此进行仔细研究。欢呼声
dreza 2012年

11

也许不是最好的dev-ops解决方案,但对于一般用于局域网或类似场合的服务器来说却很不错。

用于screen在退出之前先运行服务器,然后分离,这将使进程保持运行状态,然后可以随时重新连接。

工作流程:

开始屏幕: screen

启动服务器: java -jar minecraft-server.jar

分离按:Ctl-ad

重新连接: screen -r

此处提供更多信息:https : //www.gnu.org/software/screen/manual/screen.html





4

这是一个示例shell脚本(确保将MATH名称替换为您的应用程序名称):

#!/bin/bash

### BEGIN INIT INFO
# Provides:                 MATH
# Required-Start:           $java
# Required-Stop:            $java
# Short-Description:        Start and stop MATH service.
# Description:              -
# Date-Creation:            -
# Date-Last-Modification:   -
# Author:                   -
### END INIT INFO

# Variables
PGREP=/usr/bin/pgrep
JAVA=/usr/bin/java
ZERO=0

# Start the MATH
start() {
    echo "Starting MATH..."
    #Verify if the service is running
    $PGREP -f MATH > /dev/null
    VERIFIER=$?
    if [ $ZERO = $VERIFIER ]
    then
        echo "The service is already running"
    else
        #Run the jar file MATH service
        $JAVA -jar /opt/MATH/MATH.jar > /dev/null 2>&1 &
        #sleep time before the service verification
        sleep 10
        #Verify if the service is running
        $PGREP -f MATH  > /dev/null
        VERIFIER=$?
        if [ $ZERO = $VERIFIER ]
        then
            echo "Service was successfully started"
        else
            echo "Failed to start service"
        fi
    fi
    echo
}

# Stop the MATH
stop() {
    echo "Stopping MATH..."
    #Verify if the service is running
    $PGREP -f MATH > /dev/null
    VERIFIER=$?
    if [ $ZERO = $VERIFIER ]
    then
        #Kill the pid of java with the service name
        kill -9 $($PGREP -f MATH)
        #Sleep time before the service verification
        sleep 10
        #Verify if the service is running
        $PGREP -f MATH  > /dev/null
        VERIFIER=$?
        if [ $ZERO = $VERIFIER ]
        then
            echo "Failed to stop service"
        else
            echo "Service was successfully stopped"
        fi
    else
        echo "The service is already stopped"
    fi
    echo
}

# Verify the status of MATH
status() {
    echo "Checking status of MATH..."
    #Verify if the service is running
    $PGREP -f MATH > /dev/null
    VERIFIER=$?
    if [ $ZERO = $VERIFIER ]
    then
        echo "Service is running"
    else
        echo "Service is stopped"
    fi
    echo
}

# Main logic
case "$1" in
    start)
        start
        ;;
    stop)
        stop
        ;;
    status)
        status
        ;;
    restart|reload)
        stop
        start
        ;;
  *)
    echo $"Usage: $0 {start|stop|status|restart|reload}"
    exit 1
esac
exit 0

出于某种原因,它总是报告服务已启动。从脚本内部运行时,似乎pgrep返回0,但是如果我手动输入pgrep命令,它将返回
1。– HomeIsWhereThePcIs

pgrep认为服务正在运行的原因是因为它检测到“ / bin / sh / sbin / service MATH start”和“ / bin / bash /etc/init.d/MATH start”并返回0
HomeIsWhereThePcIs

3

Spring Boot应用程序即服务,我可以推荐基于Python的supervisord应用程序。有关更多信息,请参见该堆栈溢出问题。设置起来非常简单。


supervisord是伟大的,对于那些不知道是谁,它允许监控服务(必须是foreground-没有daemonized),它就会自动重新启动服务(可以重新启动时通过插件发生电子邮件警报)
wired00

2

根据您的平台,其他答案可以很好地提供自定义脚本和设置。除了这些,我还了解以下成熟的专用程序:

  • TanukiSoftware的JSW
  • YAJSW是上面的开源克隆。它是用Java编写的,并且是一个保姆进程,可以根据配置管理子进程(您的代码)。适用于Windows / Linux。
  • JSVC是本机应用程序。它也是一个保姆进程,但是它通过JNI而不是子进程来调用子应用程序。


1

Spring Boot参考指南

作为init.d服务安装(系统V)

简单符号链接罐子init.d,支持标准startstoprestartstatus命令。假设您在/ var / myapp中安装了Spring Boot应用程序,要将Spring Boot应用程序安装为init.d服务,只需创建一个符号链接:

$ sudo ln -s /var/myapp/myapp.jar /etc/init.d/myapp

安装后,您可以按照通常的方式启动和停止服务。例如,在基于Debian的系统上:

$ service myapp start

小费如果您的应用程序无法启动,请检查写入的日志文件中/var/log/<appname>.log是否有错误。

继续阅读以了解如何保护已部署的服务。

完成编写后,我发现我的服务无法在日志中以以下错误消息开头start-stop-daemon:无法识别的选项--no-close。并且我设法通过创建/var/myapp/myapp.conf具有以下内容的配置文件来修复它

USE_START_STOP_DAEMON=false

1

我有Netty Java应用程序,我想用systemd将其作为服务运行。不幸的是,无论我使用哪种类型,应用程序都会停止。最后,我将java start包装在屏幕中。这是配置文件:

服务

[Unit]
Description=Netty service
After=network.target
[Service]
User=user
Type=forking
WorkingDirectory=/home/user/app
ExecStart=/home/user/app/start.sh
TimeoutStopSec=10
Restart=on-failure
RestartSec=5
[Install]
WantedBy=multi-user.target

开始

#!/bin/sh
/usr/bin/screen -L -dmS netty_app java -cp app.jar classPath

从那时起,您可以使用systemctl [start|stop|status] service



0

但是一旦启动,我不知道如何访问它以停止它

您可以编写一个简单的停止脚本,该脚本会为Java进程抓紧时间,提取PID并在其上调用kill。这不花哨,但很简单。像这样的开始可能会有所帮助:

#!/bin/bash
PID = ps ax | grep "name of your app" | cut -d ' ' -f 1
kill $PID

2
我在linux上不是很好,但是不会pkill nameofprocess做同样的事情吗?
DenysSéguret2012年

0

可以将war作为Linux服务运行,并且您可能想在打包之前强行插入pom.xml文件,因为某些发行版可能在自动模式下无法识别。为此,请在spring-boot-maven-plugin插件中添加以下属性。

                    <embeddedLaunchScriptProperties>
                        <mode>service</mode>
                    </embeddedLaunchScriptProperties>

接下来,使用以下命令设置您的init.d:

ln -s myapp.war /etc/init.d/myapp

你将能够运行

service myapp start|stop|restart

您可以在Spring Boot文档中找到许多其他选项,包括Windows服务。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.