开发JAVA工程代码(main方法启动)如何构建出可以通过java -jar命令直接执行的jar?如果工程中依赖其他jar又如何处理?本文即针对以上两点通过IDEA开发工具来做一个简单的分析和尝试,测试如下四种方法,如有瑕疵请轻喷。
文字目录
- 1.通过IDEA直接构建可执行jar
- 2.通过ANT构建可执行jar
- 3.通过maven-assembly插件构建可执行jar
- 4.通过maven-shade插件构建可执行jar
1.通过IDEA直接构建可执行jar
如下所示通过IDEA创建一个本地JAVA工程,此处工程名JavaPackage1,在lib引入一个hutool-all-5.3.7.jar作为测试,目录结构如下点击运行JPackage1正常打印日期。
如何通过IDEA直接构建可执行jar呢?具体步骤如下:
-
选择File -> Project Structure
-
选择Arifacts->ADD->JAR->Form modules with dependencies
-
在Main Class栏选择要打包成jar文件的java类,点击OK
-
出现如下界面,点击ok。最后构建完成jar的启动相关信息会写入到META-INF/MANIFEST.MF清单文件中。
-
点击Build->Build Artifacts,参照以下两张图,最后在\out\artifacts\目录下生成j可执行ar文件
-
最后通过在cmd窗口直接执行java -jar JavaPackage1.jar 运行正常输出hutool工具类获取的日期
后续若修改代码后可直接执行Build->Build Artifacts生成jar,不需要再做前面的配置步骤。同时解压该生成的jar可以发现依赖的hutool工具类是以class文件全部打入进来了。其中生成的MANIFEST.MF文件很简单就指明了Main-Class启动类信息。
2.通过ANT构建可执行jar
- IDEA新建JAVA工程,此处新建工程名JavaPackage2,并在lib下引入hutool依赖jar
- 在项目根目录下编写ant.xml(文件名无要求)
xml中内容如下,具体语法可参考官网学习
<?xml version="1.0" encoding="UTF8"?>
<project name="buildJar" default="jar" basedir=".">
<tstamp>
<format property="TODAY" pattern="yyyy-MM-dd HH:mm:ss" />
</tstamp>
<!-- 定义:打包生成的jar的名字 -->
<property name="jar.name" value="JavaPackage2.jar"/>
<!-- 定义:主类路径 -->
<property name="main.class" value="com.xren.jpackage2.JPackage2"/>
<property name="build.dir" value="${basedir}/build" />
<property name="classes.dir" value="${basedir}/classes" />
<property name="lib.dir" value="${basedir}/lib" />
<!--阶段1.删除之前生成的jar包-->
<target name="clean">
<delete dir="${build.dir}"/>
<delete dir="${classes.dir}"/>
</target>
<!--定义依赖jar路径-->
<path id="lib.path">
<fileset dir="${lib.dir}">
<include name="*.jar"/>
</fileset>
</path>
<!--阶段2.编译java文件-->
<target name="compile" depends="clean">
<!--创建classes文件夹-->
<mkdir dir="${classes.dir}"/>
<!--编译java文件-->
<javac srcdir="${basedir}/src" destdir="${classes.dir}" encoding="UTF8">
<!--编译依赖jar -->
<classpath refid="lib.path"/>
</javac>
</target>
<!--阶段3.生成可运行的jar包-->
<pathconvert property="relate.classpath" pathsep=" ">
<mapper>
<chainedmapper>
<flattenmapper />
<globmapper from="*" to="lib/*" />
</chainedmapper>
</mapper>
<path refid="lib.path" />
</pathconvert>
<target name="jar" depends="compile">
<mkdir dir="${build.dir}"/>
<!--指定清单文件 -->
<manifest file="${build.dir}/MANIFEST.MF">
<!--主类路径-->
<attribute name="Main-class" value="${main.class}"/>
<!-- 引入依赖jar-classpath -->
<attribute name="Class-Path" value="${relate.classpath}" />
<attribute name="Built-Date" value="${TODAY}" />
</manifest>
<jar destfile="${build.dir}/${jar.name}"
basedir="${classes.dir}"
manifest="${build.dir}/MANIFEST.MF">
</jar>
<!--拷贝依赖jar到当前build目录 -->
<copy todir="${build.dir}/lib" >
<fileset dir="${lib.dir}"></fileset>
</copy>
</target>
</project>
- 添加ant.xml文件
- 选中buildJar、点击执行,而后便会在build目录下生成一个可执行的jar
- 执行效果如下
此方式生成的jar需要和同级目录下的lib依赖引入才能运行成功,原因是在ant.xml文件中打包时已经在清单文件MANIFEST.MF里固定了lib路径。
如果想将外部的jar全部打入到jar中,需要在xml中做如下单独引入,最后jar可直接运行。xml配置及包结构如下:
<target name="jar" depends="compile">
<mkdir dir="${build.dir}"/>
<!--指定清单文件 -->
<manifest file="${build.dir}/MANIFEST.MF">
<!--主类路径-->
<attribute name="Main-class" value="${main.class}"/>
</manifest>
<jar destfile="${build.dir}/${jar.name}"
basedir="${classes.dir}"
manifest="${build.dir}/MANIFEST.MF">
<zipfileset src="${lib.dir}/hutool-all-5.3.7.jar"/>
</jar>
</target>
3.通过maven-assembly插件构建可执行jar
- 新建Java-maven工程,通过pom引入hutool工具库依赖,此处新建工程名JavaPackage3
pom.xml配置
<dependencies>
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.3.7</version>
</dependency>
</dependencies>
如果hutool-all.jar依赖下载不了请检查IDEA的maven配置是否正确。
- Pom中引入maven-assembly-plugin插件打包
<build>
<plugins>
<!-- 使用maven-assembly-plugin插件打包 -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-assembly-plugin</artifactId>
<version>3.2.0</version>
<configuration>
<archive>
<manifest>
<!--主类 -->
<mainClass>com.xren.jpackage3.JPackage3</mainClass>
</manifest>
</archive>
<descriptorRefs>
<!-- 可执行jar名称结尾-->
<descriptorRef>jar-with-dependencies</descriptorRef>
</descriptorRefs>
</configuration>
<executions>
<execution>
<id>make-assembly</id>
<phase>package</phase>
<goals>
<goal>single</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
- 执行maven-package,而后在target目录下会生成一个以jar-with-dependencies结尾的可直接执行jar,另外一个jar里面没有依赖不可执行
- 该可执行jar的目录结构和清单文件MANIFEST.MF内容如下
通过Maven-assembly插件构建的jar和ant构建出来的jar中结构基本一致,将依赖jar以class方式加入内容部,同时在打包配置文件中指定清单文件MANIFEST.MF的主启动类Main-Class。
4.通过maven-shade插件构建可执行jar
- 新建Java-maven工程,通过pom引入hutool工具库依赖,此处新建工程名JavaPackage4
- Pom文件中引入maven-shade-plugin插件
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>3.2.4</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<transformers>
<transformer
implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>com.xren.jpackage4.JPackage4</mainClass>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
- 执行maven package,而后会在target目录下生成一个可执行JavaPackage4-1.0-SNAPSHOT.jar
- 执行该jar,正常运行
利用maven-shade-plugin构建除了会生成可执行jar,还会在target目录下生成一个以original-开头的原始jar,该jar不可执行内部没有依赖其他,其清单文件MANIFEST.MF如下:
而可执行jar清单文件如下,差别就是多了一个Main-Class
**总结:**综上几种构建可执行jar的方式可知,其都有一个共同点-jar里面有一个包含Main-Class的MANIFEST.MF文件,Main-Class配置指向main方法的主启动类。jar依赖的其他程序会以lib-jar或者class文件的方式包含其中(此文只验证依赖jar,没有验证配置文件依赖)。