java jar包后台运行方式
zoyation 2024-08-07 15:05:02 阅读 89
在实际工作中,java开发的spring boot等通过jar包部署需要一直运行的程序部署到服务器上时,都希望后台运行,方便管理程序服务、防止被误操作关闭,本文结合自己工作经验讲解jar包后台运行的两种方式,分别是按操作系统支持的特殊方式和统一执行命令的方式。
方式一:按操作系统支持的方式后台运行
可执行jar包程序可以按操作系统支持的方式运行,不同操作系统执行命令和方式不一样,这里主要讲解linux操作系统和window操作系统下如何按操作系统支持的特殊方式后台运行。
linux操作系统
Linux操作系统java程序后台运行又主要分如下两种方式:
1.通过nohup命令和&符号运行。
终端关闭后程序也会继续运行,示例如下:
<code>nohup java -jar demo.jar > nohup.log 2>&1 &
示例命令说明:
nohup:使得终端关闭,运行的命令也不中断。
java -jar demo.jar:用于启动jar包。
nohup.log:标准输出重定向到nohup.log文件。
2>&1:标准错误重定向到标准输出(即nohup.log文件)。
&:命令放入后台执行。
执行上述命令后,程序后台运行,日志记录到nohup.log里,可以使用tail等命令看日志文件,并且会得到一个进程ID(PID,这个PID可以通过ps ax|grep "demo.jar"查找),可以使用kill命令通过这个PID来终止进程。
2.jar配置为可自启动的服务。
在Linux上将jar文件设置为服务需要编写一个系统服务单元文件(.service文件),然后使用systemd来管理服务。以下是一个示例:
创建服务单元文件 <code>/etc/systemd/system/your-service.service:
[Unit]
Description=Your Java Application as a Service
After=network.target
[Service]
User=<username>
Type=simple
ExecStart=/usr/bin/java -jar /path/to/your-application.jar
Restart=on-failure
[Install]
WantedBy=multi-user.target
重新加载systemd管理器配置:
sudo systemctl daemon-reload
启动服务:
sudo systemctl start your-service.service
设置服务开机自启:
sudo systemctl enable your-service.service
确保替换 <username>
和 /path/to/your-application.jar
为实际的用户名和jar文件路径。如果需要传递额外的Java选项,可以在 ExecStart
中添加。
请注意,这个示例假定你已经有权限执行systemctl命令,并且你的系统已经安装了Java运行时环境(JRE)或Java Development Kit(JDK)。如果没有安装Java,你需要先安装它,通常可以使用系统的包管理器,例如在Ubuntu上使用 sudo apt-get install default-jdk
。
window操作系统
要在Windows环境下使jar包在后台运行,可以使用javaw命令代替java命令,javaw命令不会打开命令行窗口,适合运行没有图形界面但需要在后台运行的Java应用程序;也可以使用winsw
工具安装为系统服务实现后台运行。
结合操作系统支持的方式,window下java程序后台运行又主要分如下四种方式:
1.使用javaw -jar demo.jar命令运行jar包
打开命令提示符或PowerShell,使用javaw -jar demo.jar命令运行jar包。
2.定时运行或开机自启
如果需要定时运行或开机自启,可以创建一个批处理文件来运行jar包,可以通过Windows任务计划程序来设置。
示例批处理文件(run-jar.bat):
@echo off
start javaw -jar demo.jar
exit
3.通过VBS脚本运行
如果你想要该进程在后台默默运行,可以创建VBS脚本来启动它:
CreateObject("Wscript.Shell").Run "cmd /c start javaw -jar your-application.jar", 0, True
保存为.vbs文件,例如run-jar.vbs,双击该文件即可在后台运行jar包。
4.安装为服务方式运行
将其作为服务安装,可以使用winsw
工具来将你的Java应用程序包装成Windows服务。这样可以确保即使命令行关闭,程序也会继续运行。
下载winsw的二进制文件,并重命名为你的服务名称,例如demo.exe。然后创建一个配置文件demo.xml,最后通过install安装为windows服务。
安装服务的基本步骤如下:
下载winsw二进制文件。重命名为demo.exe。创建demo.xml配置文件,指定jar路径和其他参数。运行demo.exe install来安装服务。这样,你的Java应用程序就会作为Windows服务在后台运行,并可以设置为开机启动。
方式二:统一执行命令的方式(Runtime.getRuntime().exec())
方式一虽然都能实现jar程序后台运行但是方式并不统一,不同操作系统需要按不同方式配置,java程序既然是跨平台的,为什么个不能让jar包后台运行的命令也统一呢?答案是肯定可以的,这儿要讲的就是这种方式,实现也比较简单,Vertx等项目就是按这种方式,通过java 调用系统命令实现(Runtime.getRuntime().exec())。
这种方式目标
使用相同启动命令使得程序可以在不同操作系统上后台运行。
实现思路(Spring boot项目举例)
程序启动main方法传入一个后台运行的标志参数,比如为start。main方法里获取args参数判断是否包含start参数,如果包含就获取启动命令和按操作系统设置其他启动参数,然后通过java 调用系统命令方式(Runtime.getRuntime().exec())再次执行不包括start参数的启动命令。通过java 调用系统命令方式(Runtime.getRuntime().exec())再次执行不包括start参数的启动命令后,又进入main方法,main方法args参数里没有start参数,就执行Spring boot项目启动代码,这样就实现了程序后台运行。
执行命令示例:
java -jar demo.jar start
示例代码如下:
<code>package com.demo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import java.io.File;
import java.util.*;
@SpringBootApplication
public class DemoApplication {
public static final String START = "start";
private static String osName = System.getProperty("os.name").toLowerCase();
private static String id = UUID.randomUUID().toString();
public static void main(String[] args) {
HashSet<String> argSet = new HashSet<>(Arrays.asList(args));
if (argSet.contains("-Dapplication.id")||!argSet.contains(START)) {
SpringApplication.run(DemoApplication.class, args);
} else{
start();
}
}
/**
* 参考io.vertx.core.impl.launcher.commands.StartCommand
*/
public static void start() {
System.out.println("Starting application background...");
List<String> cmd = new ArrayList<>();
ProcessBuilder builder = new ProcessBuilder();
addJavaCommand(cmd);
// Add the classpath to env.
builder.environment().put("CLASSPATH", System.getProperty("java.class.path"));
//jar包运行
if (getJar() != null) {
cmd.add("-jar");
cmd.add(getJar());
} else {
//开发工具运行
cmd.add(getFirstSegmentOfCommand());
cmd.add("run");
}
cmd.add("-Dapplication.id=" + id);
try {
builder.command(cmd);
builder.start();
System.out.println(id);
} catch (Exception e) {
System.out.println("Cannot create application process");code>
System.exit(12);
}
}
private static void addJavaCommand(List<String> cmd) {
if (osName.contains("windows")) {
cmd.add("cmd.exe");
cmd.add("/C");
cmd.add("start");
cmd.add("application-id - " + id);
cmd.add("/B");
}
cmd.add(getJava().getAbsolutePath());
String opts = System.getenv("JAVA_OPTS");
if (opts != null) {
cmd.addAll(Arrays.asList(opts.split(" ")));
}
}
private static File getJava() {
File java;
File home = new File(System.getProperty("java.home"));
if (osName.contains("windows")) {
java = new File(home, "bin/java.exe");
} else {
java = new File(home, "bin/java");
}
if (!java.isFile()) {
System.out.println("Cannot find java executable - " + java.getAbsolutePath() + " does not exist");
System.exit(14);
}
return java;
}
public static String getJar() {
// Check whether or not the "sun.java.command" system property is defined,
// if it is, check whether the first segment of the command ends with ".jar".
String segment = getFirstSegmentOfCommand();
if (segment != null && segment.endsWith(".jar")) {
return segment;
} else {
// Second attend is to check the classpath. If the classpath contains only one element,
// it's the fat jar
String classpath = System.getProperty("java.class.path");
if (!classpath.isEmpty() && !classpath.contains(File.pathSeparator) && classpath.endsWith(".jar")) {
return classpath;
}
}
return null;
}
/**
* @return the first segment of the command line.
*/
public static String getFirstSegmentOfCommand() {
String cmd = System.getProperty("sun.java.command");
if (cmd != null) {
String[] segments = cmd.split(" ");
if (segments.length >= 1) {
return segments[0];
}
}
return null;
}
}
声明
本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。