美文网首页Java高级进阶
Maven把项目依赖的所有jar包都打到同一个jar中

Maven把项目依赖的所有jar包都打到同一个jar中

作者: java菜 | 来源:发表于2019-03-30 14:28 被阅读2次

    1 使用maven-shade-plugin

    (1) 在项目的pom.xml文件中加入如下插件:

        <build>

            <plugins>

                <!-- Maven Shade Plugin -->

                <plugin>

                    <groupId>org.apache.maven.plugins</groupId>

                    <artifactId>maven-shade-plugin</artifactId>

                    <version>3.2.0</version>

                    <configuration>

                        <createDependencyReducedPom>true</createDependencyReducedPom>

                    </configuration>

                    <executions>

                        <execution>

                            <phase>package</phase>

                            <goals>

                                <goal>shade</goal>

                            </goals>

                            <configuration>

                                <transformers>

                                    <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">

                                        <!-- 主类的全限定名 -->

                                        <mainClass>com.healchow.consumer.Main</mainClass>

                                    </transformer>

                                </transformers>

                            </configuration>

                        </execution>

                    </executions>

                </plugin>

            </plugins>

        </build>

    (2) 然后使用Maven的package命令打包项目;

    (3) 打包完成后, 即可在项目的target目录下看到生成的jar包;

    (4) 不出差错的话, 使用java -jar xxxx.jar命令即可运行此jar包.

    说明: 此方法在Mac OS系统下的IDEA中实施, 在jar/META-INF/目录下有时会出现:

    ECLIPSEF.SF ECLIPSE.RSA等文件, 并在运行java -jar xxx.jar时, 会抛出如下错误:

    Error: A JNI error has occurred, please check your installation and try again

    Exception in thread "main" java.lang.SecurityException: Invalid signature file digest for Manifest main attributes

            at sun.security.util.SignatureFileVerifier.processImpl(SignatureFileVerifier.java:330)

            at sun.security.util.SignatureFileVerifier.process(SignatureFileVerifier.java:263)

            at java.util.jar.JarVerifier.processEntry(JarVerifier.java:318)

            at java.util.jar.JarVerifier.update(JarVerifier.java:230)

            at java.util.jar.JarFile.initializeVerifier(JarFile.java:383)

            at java.util.jar.JarFile.getInputStream(JarFile.java:450)

            at sun.misc.URLClassPath$JarLoader$2.getInputStream(URLClassPath.java:977)

            at sun.misc.Resource.cachedInputStream(Resource.java:77)

            at sun.misc.Resource.getByteBuffer(Resource.java:160)

            at java.net.URLClassLoader.defineClass(URLClassLoader.java:454)

            at java.net.URLClassLoader.access$100(URLClassLoader.java:73)

            at java.net.URLClassLoader$1.run(URLClassLoader.java:368)

            at java.net.URLClassLoader$1.run(URLClassLoader.java:362)

            at java.security.AccessController.doPrivileged(Native Method)

            at java.net.URLClassLoader.findClass(URLClassLoader.java:361)

            at java.lang.ClassLoader.loadClass(ClassLoader.java:424)

            at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:335)

            at java.lang.ClassLoader.loadClass(ClassLoader.java:357)

            at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:495)

    解决办法是: 手动删除jar包中META-INF/*.RSA META-INF/*.DSA META-INF/*.SF这些文件, 然后就可行了.

    2 推荐: 使用maven-assembly-plugin

    (1) 在项目的pom.xml文件中加入如下插件:

        <build>

            <plugins>

                <!-- Maven Assembly Plugin -->

                <plugin>

                    <groupId>org.apache.maven.plugins</groupId>

                    <artifactId>maven-assembly-plugin</artifactId>

                    <version>2.4.1</version>

                    <configuration>

                        <!-- get all project dependencies -->

                        <descriptorRefs>

                            <descriptorRef>jar-with-dependencies</descriptorRef>

                        </descriptorRefs>

                        <!-- MainClass in mainfest make a executable jar -->

                        <archive>

                            <manifest>

                                <mainClass>com.healchow.consumer.Main</mainClass>

                            </manifest>

                        </archive>

                    </configuration>

                    <executions>

                        <!-- 配置执行器 -->

                        <execution>

                            <id>make-assembly</id>

                            <!-- 绑定到package命令的生命周期上 -->

                            <phase>package</phase>

                            <goals>

                                <!-- 只运行一次 -->

                                <goal>single</goal>

                            </goals>

                        </execution>

                    </executions>

                </plugin>

            </plugins>

        </build>

    (2) 然后使用Maven的package命令打包项目;

    (3) 打包完成后, 即可在项目的target目录下看到生成的jar包, 其中xxx-jar-with-dependencies.jar是包含依赖jar包的jar文件, 另一个不包含依赖jar包;

    (4) 使用java -jar xxxx.jar命令即可运行此jar包.

    3 扩展: Maven安装本地jar包到本地仓库

    命令如下:

    mvn install:install-file -Dfile=base-util-1.0.RELEASE.jar -DgroupId=com.healchow -DartifactId=base-util -Dversion=1.0.RELEASE -Dpackaging=jar

    其他命令, 参考这篇博客: Maven常用命令.

    4 扩展: 手动生成jar包

    创建可执行 jar 包的关键在于:让java -jar命令知道 jar 包中的主方法属于哪个类 —— 要配置主类的全限定名.

    步骤如下:

    (1) 打包指定的项目:

    进入项目所在的根目录, 以Linux系统为例:

    项目名是java-test, 其所在的目录是/data/project/java-test;main方法所属的类的全限定名是com.healchow.test.Main:

    # 进入项目所在的根目录:

    cd /data/project

    # 运行打包命令:

    jar  cvfe  test.jar  java-test.com.healchow.test.Main  java-test

    上述命令说明: 将java-test目录下的所有文件打包到test.jar中, 并指定主类的全限定名称, 这里要指明主类所在的文件夹.

    关于jar命令的其他用法, 参考如下:

    # 用法: jar {ctxui}[vfmn0PMe] [jar-file] [manifest-file] [entry-point] [-C dir] files ...

    # 选项:

    -c  创建新档案

    -t  列出档案目录

    -x  从档案中提取指定的 (或所有) 文件

    -u  更新现有档案

    -v  在标准输出中生成详细输出

    -f  指定档案文件名

    -m  包含指定清单文件中的清单信息

    -n  创建新档案后执行 Pack200 规范化

    -e  为捆绑到可执行 jar 文件的独立应用程序指定应用程序入口点

    -0  仅存储; 不使用任何 ZIP 压缩

    -P  保留文件名中的前导 '/' (绝对路径) 和 ".." (父目录) 组件

    -M  不创建条目的清单文件

    -i  为指定的 jar 文件生成索引信息

    -C  更改为指定的目录并包含以下文件

    # 如果任何文件为目录, 则对其进行递归处理。

    # 清单文件名, 档案文件名和入口点名称的指定顺序 与 'm', 'f' 和 'e' 标记的指定顺序相同。

    示例 1: 将两个类文件归档到一个名为 classes.jar 的档案中:

        jar cvf classes.jar Foo.class Bar.class

    示例 2: 使用现有的清单文件 'mymanifest' 并将 foo/ 目录中的所有文件归档到 'classes.jar' 中:

        jar cvfm classes.jar mymanifest -C foo/ .

    (2) 运行jar包:

    用解压软件打开上一步生成的jar包, 查看META-INF/MANIFEST.MF文件, 里面如果有Main-Class: com.healchow.test.Main, 说明打包成功.

    # 在当前目录下, 运行jar包:

    java -jar test.jar

    5 扩展: Linux下运行jar包的几种方式

    5.1 阻塞式方式

    java -jar xxx.jar

    特点: 当前终端的会话窗口被阻塞, 可按CTRL + C中断程序的运行, 或者直接关闭会话窗口 —— 此时jar程序将终止.

    5.2 后台运行方式

    # 通过"&"符号实现后台运行:

    java -jar xxx.jar &

    特点: 当前终端的会话窗口不会被阻塞, 但是如果关闭当前窗口, 程序将终止运行.

    5.3 后台持续运行方式

    # 通过"nohup"命令实现后台运行:

    nohup java -jar xxx.jar &

    nohup命令是不挂断运行命令的意思.

    特点: 当用户退出或终端关闭时, 程序的运行不受影响.

    说明: 用nohup命令运行任务时, 默认情况下这个任务的所有输出都将被重定向到nohup.out文件中, 除非另外指定输出文件. 指定输出文件方式为:

    # 将输出重定向到xxx.log文件中:

    nohup java -jar xxx.jar > xxx.log &

    > 符号是重定向符, 实现: 把它之前命令的所有输出信息重定向到后面的文件中, 而不是显示到屏幕上.

    5.4 其他命令扩展

    ① 通过jobs命令可以查看后台运行任务, 会给每个任务指定一个编号.

    ② 通过fg命令可以把指定编号的后台任务调回到前台控制, 如: fg 10 —— 把10号任务调回前台.

    ③ 查看占用某个端口的进程:

    netstat -nlp | grep 2181

    # 下述结果说明进程号pid=18434的java进程占用了2181端口

    tcp        0      0 :::2181          :::*          LISTEN      18434/java

    欢迎工作一到五年的Java工程师朋友们加入Java程序员开发: 721575865

    群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)合理利用自己每一分每一秒的时间来学习提升自己,不要再用"没有时间“来掩饰自己思想上的懒惰!趁年轻,使劲拼,给未来的自己一个交代!

    相关文章

      网友评论

        本文标题:Maven把项目依赖的所有jar包都打到同一个jar中

        本文链接:https://www.haomeiwen.com/subject/hrnjbqtx.html