背景

现在很多软件都支持集群部署,但是测试环境通常资源有限,所以一般通过单台机器模拟集群部署(使用不同端口,运行相同jar包),本文的目的就是通过多种方式实现此需求。

两个程序

1、jar程序

  ① springboot程序

  ② 只包含一个main方法,用于启动程序,输出进程ID

  ③ 路径:C:/demo.jar(windows) /demo.jar(Linux)

2、启动程序

  ① 包含main方法的程序

多种方式

1、通过URLClassLoader加载jar程序(windows平台)

2、通过java -jar命令启动jar程序(windows平台)

3、通过复制原始jar文件,启动不同的jar程序(windows平台)

4、通过Linux Shell脚本启动(Linux平台)

方式一

1、通过URLClassLoader加载jar程序(windows平台)

  ① 说明

    1) 启动程序多次加载jar程序

    2) jar程序和启动程序使用相同进程,非独立进程,无实际意义,仅介绍

  ② 启动jar程序:运行启动程序main方法

  ③ 终止jar程序:停止启动程序(因为共用同一个进程,终止主程序,jar程序会同时终止)

2、代码

  ① jar程序

@SpringBootApplication

public class DemoStarter {

    public static void main(String[] args) {

        // 获取进程Id

        String name = ManagementFactory.getRuntimeMXBean().getName();

        String processId = name.split("@")[0];

        System.out.println(processId);

        SpringApplication.run(DemoStarter.class, args);
}
}

  ② 启动程序

import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader; public class Starter1 { public static void main(String[] args) throws Exception { start("7001");
start("7002");
start("7003");
} private static void start(String port) throws Exception { String path = "file:" + "C:/demo.jar"; URLClassLoader classLoader = new URLClassLoader(new URL[]{new URL(path)}); // jar程序的启动类完整路径
Class demo = classLoader.loadClass("DemoStarter"); Method method = demo.getMethod("main", String[].class); method.invoke(null, (Object) new String[]{port});
}
}

方式二

1、通过java -jar命令启动jar包(windows平台)

  ① 说明

    1) 启动程序使用命令多次启动jar包

    2) 动态构建cmd命令(不同参数),启动相同jar程序,各个jar程序使用不同进程

  ② 启动jar程序

    1) 运行启动程序main方法

    2) 保存各个进程ID到文件

  ③ 终止jar程序

    1) 根据保存的进程ID停止各个jar程序

2、代码

  ① jar程序(同方式一)

  ② 启动程序

import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader; public class Starter2 { public static void main(String[] args) throws Exception { cmd("7001");
cmd("7002");
cmd("7003"); // 根据文件中的进程Id终止程序
killByProcessId("PID1");
killByProcessId("PID2");
killByProcessId("PID3");
} private static void cmd(String port) throws Exception { String cmd = "java -jar -Dserver.port=" + port + " " + "C:/demo.jar"; Process p = Runtime.getRuntime().exec(cmd);
InputStream is = p.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(is)); // 获取进程Id(DemoStarter-->main方法
// reader.readLine()第一行为System.out.println(processId)输出内容
String processId;
while ((processId = reader.readLine()) != null) {
break;
}
is.close();
reader.close(); // 这里可以将进程ID保存到文件中
System.out.println("processId:" + processId);
} private static void killByProcessId(String processId) throws Exception { String cmd = "taskkill /F /PID \"" + processId + "\"";
Runtime.getRuntime().exec(cmd);
}
}

方式三

1、通过复制原始jar文件,启动不同的jar程序(windows平台)

  ① 说明

    1) 复制原始jar包,生成新的jar包

    2) 动态构建cmd命令(不同参数、不同jar包名称),启动不同jar包,各个jar包使用不同进程

  ② 启动jar程序

    1) 运行启动程序的main方法

    2) 保存各个进程ID到文件

  ③ 终止程序

    1) 根据保存的进程ID停止各个jar程序

  ④ 复制jar文件需要时间,但可以解决启动相同jar包可能存在的问题

2、代码

  ③ jar程序(同方式一)

  ④ 启动程序

import org.apache.commons.io.FileUtils;

import java.io.BufferedReader;
import java.io.File;
import java.io.InputStream;
import java.io.InputStreamReader; public class Starter3 { public static void main(String[] args) throws Exception { copyCmd(); // 根据文件中的进程Id终止程序
killByProcessId("PID1");
killByProcessId("PID2");
killByProcessId("PID3");
} private static void copyCmd() throws Exception { File srcFile = new File("C:/demo.jar");
File destiFile2 = new File("C:/demo2.jar");
File destiFile3 = new File("C:/demo3.jar"); // 删除之前复制的jar包
FileUtils.forceDelete(destiFile2);
FileUtils.forceDelete(destiFile3); FileUtils.copyFile(srcFile, destiFile2);
FileUtils.copyFile(srcFile, destiFile3); copy("java -jar -Dserver.port=7001 C:/demo.jar");
copy("java -jar -Dserver.port=7002 C:/demo2.jar");
copy("java -jar -Dserver.port=7003 C:/demo3.jar");
} private static void copy(String cmd) throws Exception { Process p; p = Runtime.getRuntime().exec(cmd);
InputStream is = p.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(is)); // 获取进程Id(DemoStarter-->main方法
// reader.readLine()第一行为System.out.println(processId)输出内容
String processId;
while ((processId = reader.readLine()) != null) {
break;
}
is.close();
reader.close(); // 这里可以将进程ID保存到文件中
System.out.println("ProcessId:" + processId);
} private static void killByProcessId(String processId) throws Exception { String cmd = "taskkill /F /PID \"" + processId + "\"";
Runtime.getRuntime().exec(cmd);
}
}

方式四

1、通过Linux Shell脚本启动(Linux平台)

  ① 执行java -jar命令

  ② 根据端口获取进程ID

  ③ 根据进程ID终止程序

2、代码

  ① jar程序(同方式一)

  ② Shell命令

    1) 启动程序

#!/bin/bash

java -jar -Dserver.port=7001 /demo.jar
java -jar -Dserver.port=7002 /demo.jar
java -jar -Dserver.port=7003 /demo.jar

  2) 终止程序

#!/bin/bash

pid1=`netstat -anp | grep 7001 | awk '{printf $7}' | cut -d/ -f1`
pid2=`netstat -anp | grep 7002 | awk '{printf $7}' | cut -d/ -f1`
pid3=`netstat -anp | grep 7003 | awk '{printf $7}' | cut -d/ -f1` kill ${pid1}
kill ${pid2}
kill ${pid3}

问题&总结

  1、方式一可以通过调用method.invoke传递参数

  2、其它方式可通过jar命令传递参数

  3、启动程序通过Process启动jar程序并获取jar程序进程ID

  4、多次启动jar程序时报错:”unable to register MBean” 
      设置参数spring.jmx.enabled=false

  5、根据端口号获取进程ID(windows)
    netstat -ano|findstr "7001 7002 7003"

  6、根据进程ID停止进程(windows)

    taskkill /F /PID "1"

参考资料

  1、https://my.oschina.net/u/2971292/blog/2960777

  2、https://www.jianshu.com/p/3eea5e7e1e6f

  3、https://www.cnblogs.com/sxdcgaq8080/p/10579073.html

Java多次启动相同jar程序的更多相关文章

  1. shell脚本批量/单独启动、停止、重启java独立jar程序

    本人最近半年使用阿里dubbo做开发,并在公司内部大力进行推广,将原来一个笨重且不易于维护的大项目切分成多个相对独立的java程序,好处是显而易见的,但是随着切分的独立运行程序包越来越多,程序的部署变 ...

  2. Java命令行启动jar包更改默认端口以及配置文件的几种方式

    Java命令行启动jar包更改默认端口以及配置文件的几种方式 java -jar xxx.jar --server.port=8081 默认如果jar包没有启动文件,可以采用这种方式进行启动 java ...

  3. 使用 Java Service Wrapper 启动java后台进程服务

    Java Service Wrapper (http://wrapper.tanukisoftware.com/doc/english/product-overview.html)可以很方便得在各个平 ...

  4. 手把手教你如何把java代码,打包成jar文件以及转换为exe可执行文件

    1.背景: 学习java时,教材中关于如题问题,只有一小节说明,而且要自己写麻烦的配置文件,最终结果却只能转换为jar文件.实在是心有不爽.此篇博客教你如何方便快捷地把java代码,打包成jar文件以 ...

  5. Java之Property-统获取一个应用程序运行的次数

    package FileDemo; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStre ...

  6. java classpath import package 机制 @Java的ClassPath, Package和Jar

    java classpath import package 机制   從一個簡單的例子談談package與import機制 基本原則:為什麼需要將Java文件和類文件切實安置到其所歸屬之Package ...

  7. Intellij IDEA下导出Java工程的可运行JAR包

    Intellij IDEA下导出Java工程的可运行JAR包 昨天一直向导出一个Java工程的可运行JAR包,然后查阅网上的资料以及自己一遍一遍的尝试,均以失败告终.可以导出JAR包,但是导出的JAR ...

  8. Jar程序使用MyBatis集成阿里巴巴druid连接池

    在写jar程序,而不是web程序的时候,使用mybatis作为持久层,可以集成POOLED连接池,而阿里的druid不能用,确实很郁闷.不过有办法. 首先准备好数据库配置文件 然后对Druid进行一个 ...

  9. java.io.FileNotFoundException: ..\lib\commons-el.jar

    安装openfire成功后,启动遇到java.io.FileNotFoundException: ..\lib\commons-el.jar错误,并不是缺少了jar包,只需以管理员身份运行即可解决.

随机推荐

  1. psexec与wmi在内网渗透的使用

    psexec是一个很好的管理工具,在内网渗透中也被广泛使用. 但太“出名”也往往会遭来各种麻烦. 在有安全监听.防护的内网中使用psexec会容易触发告警. 1.psexec用法(前提:对方要开启ad ...

  2. 关于logging模块

    from logging.handlers import TimedRotatingFileHandle #日志文件控制(日志删除时间设置) import logging logger=logging ...

  3. Java变量常量声明和定义

    一.常量和变量 1.常量变量定义 在程序中存在大量的数据来代表程序的状态,其中有些数据在程序的运行过程中值会发生改变,有些数据在程序运行过程中值不能发生改变,这些数据在程序中分别被叫做变量和常量. 2 ...

  4. Win10下免安装版MySQL5.7的安装和配置

    1.MySQL5.7解压 2.新建配置文件my.ini放在D:\Free\mysql-5.7.26-winx64目录下 [mysql] # 设置mysql客户端默认字符集 default-charac ...

  5. jQuery遍历之find()

    /**案例说明: *首先理清楚find()函数同children()函数之间的区别 * 1. find()会遍历给定节点下的所有的元素节点. * 2. children()之后遍历给定节点下的单一层级 ...

  6. JavaSE理论篇

    将已学过的知识记录在此,既能便于以后温习又能方便知识共享,做到共同成长. 计算机语言发展简史 主要分为三个阶段 机器语言:打点机,有点表示1,没有表示0,打点计时器 低级语言:汇编语言 高级语言:Ja ...

  7. 分布式限流组件-基于Redis的注解支持的Ratelimiter

    原文:https://juejin.im/entry/5bd491c85188255ac2629bef?utm_source=coffeephp.com 在分布式领域,我们难免会遇到并发量突增,对后端 ...

  8. 24、python re正则表达式模块

    一.re模块的基本使用 Python里数量词默认是贪婪的,总是尝试匹配尽可能多的字符.正则表达式是用来匹配处理字符串的. 假如你需要匹配文本中的字符\,Python里的使用原生字符串表示:r'\\'表 ...

  9. 【Spark】ScalaIDE运行spark,A master URL must be set in your configuration

    or SparkSession.master("local")

  10. hive的常用函数工作总结

    1.concat_ws 它是一个特殊形式的 CONCAT() concat_ws(分隔符,参数1,参数2.......) as 字段 2.split 返回值为一个数组 a.基本用法: 例1:split ...