Spring Boot 项目瘦身指南,瘦到难以想象!129M->1.3M

以前在 从使用传统Web框架到切换到Spring Boot后的总结 中提到关于 Spring Boot 编译打包,Spring Boot 应用程序不用额外部署到外部容器中,能够直接经过 Maven 命令将项目编译成可执行的 jar 包,而后经过 java -jar 命令启动便可,很是方便。html

最近有小伙伴私信我说,打 jar 包方即是方便,就是每次打包出来的 jar 太大了,先不说上传时间的问题,若是只修改了 1 个类就须要从新打包项目,而后从新上传项目到服务器,怎么以为还不如我以前使用 war 包方便呢,使用 war 包时,虽然要部署到 Tomcat 中,但只须要将修改的 class 替换一下,重启一下 Tomcat 就能够了。。。java

其实到底选择哪一种打包方式,主要仍是看我的习惯和业务场景需求,毕竟 Spring Boot 也支持打包 war 包的。web

今天的重点不是打包方式,而是解决困惑了小伙伴打包的 jar 太大的问题。面试

正常打包项目

给 Spring Boot 打包你们应该很熟了吧,只须要在 pom.xml 文件中配置 spring-boot-maven-plugin 打包插件:spring

<build>
    <plugins>
        <plugin>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-maven-plugin</artifactId>
        </plugin>
    </plugins>
</build>

而后在项目根目录执行 mvn clean pavkage 就能够完成打包了,以下是我本地的一个项目打包状况:数据库

能够看到打包出的 jar 应用是至关的大了,如小伙伴说的同样,若是每次修改一个 class 文件或者配置文件,就须要从新打包而后上传服务器的话,那确实是太麻烦了,可能上传就浪费大部分时间。。。apache

应用瘦身(分离lib和配置文件)

其实 jar 包大的缘由在于全部的依赖包所有集成在 jar 包里面,以下是瘦身前的 jar 包内部结构:服务器

其中 classes 就是咱们项目的代码,仅仅1.3M,而 129MB 的 lib 目录是项目中全部的依赖(好比spinrg、Hibernate等依赖),若是咱们能把这个 lib 目录提取出来,整个项目就会变得特别小了。说干就干。网络

咱们知道 Spring Boot 的打包终究是依赖于 Maven ,因此想到更改打包信息,无非就是指定 Maven 的配置。mybatis

在 pom.xml 添加以下信息(后文解释):

<build>
   <finalName>你想要的jar包名称</finalName>
    <plugins>
      <!-- 一、编译出不带 lib 文件夹的Jar包 -->
        <plugin>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-maven-plugin</artifactId>
            <configuration>
                <!--表示编译版本配置有效-->
                <fork>true</fork>
                <!--引入第三方jar包时,不添加则引入的第三方jar不会被打入jar包中-->
                <includeSystemScope>true</includeSystemScope>
                <!--排除第三方jar文件-->
                <includes>
                    <include>
                        <groupId>nothing</groupId>
                        <artifactId>nothing</artifactId>
                    </include>
                </includes>
            </configuration>
            <executions>
                <execution>
                    <goals>
                        <goal>repackage</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    
    <!-- 二、完成对Java代码的编译,能够指定项目源码的jdk版本,编译后的jdk版本,以及编码 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
              <!-- 源代码使用的JDK版本 --> 
                <source>${java.version}</source>
                <!-- 须要生成的目标class文件的编译版本 -->
                <target>${java.version}</target>
                <!-- 字符集编码 -->
                <encoding>UTF-8</encoding>
                <!-- 用来传递编译器自身不包含可是却支持的参数选项 -->  
                <compilerArguments>
                    <verbose/>
                    <!-- windwos环境(二选一) -->
                    <bootclasspath>${java.home}/lib/rt.jar:${java.home}/lib/jce.jar</bootclasspath>
                    <!-- Linux环境(二选一) -->
                    <bootclasspath>${java.home}/lib/rt.jar:${java.home}/lib/jce.jar</bootclasspath>
                </compilerArguments>
            </configuration>
        </plugin>

        <!-- 三、将全部依赖的jar文件复制到target/lib目录 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-dependency-plugin</artifactId>
            <executions>
                <execution>
                    <id>copy-dependencies</id>
                    <phase>prepare-package</phase>
                    <goals>
                        <goal>copy-dependencies</goal>
                    </goals>
                    <configuration>
                      <!--复制到哪一个路径,${project.build.directory} 缺醒为 target,其余内置参数见下面解释-->
                        <outputDirectory>${project.build.directory}/lib</outputDirectory>
                        <overWriteReleases>false</overWriteReleases>
                        <overWriteSnapshots>false</overWriteSnapshots>
                        <overWriteIfNewer>true</overWriteIfNewer>
                    </configuration>
                </execution>
            </executions>
        </plugin>
        
    <!-- 四、指定启动类,指定配置文件,将依赖打成外部jar包 -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-jar-plugin</artifactId>
            <configuration>
                <archive>
                    <manifest>
                        <!-- 是否要把第三方jar加入到类构建路径 -->
                        <addClasspath>true</addClasspath>
                        <!-- 外部依赖jar包的最终位置 -->
                        <classpathPrefix>lib/</classpathPrefix>
                        <!-- 项目启动类 -->
                        <mainClass>com.javam4.MyApplication</mainClass>
                    </manifest>
                </archive>
                <!--资源文件不打进jar包中,作到配置跟项目分离的效果-->
                <excludes>
                    <!-- 业务jar中过滤application.properties/yml文件,在jar包外控制 -->
                    <exclude>*.properties</exclude>
                    <exclude>*.xml</exclude>
                    <exclude>*.yml</exclude>
                </excludes>
            </configuration>
        </plugin>

    </plugins>
</build>

以下一一细拆如上配置:

一、spring-boot-maven-plugin

Springboot 默认使用 spring-boot-maven-plugin 来打包,这个插件会将项目全部的依赖打入项目 jar 包里面,正常打包时 spring-boot-maven-plugin 结构以下:

<plugin>  
    <groupId>org.springframework.boot</groupId>   
    <artifactId>spring-boot-maven-plugin</artifactId>  
    <configuration>  
        <mainClass>com.javam4.MyApplication</mainClass>  
        <layout>ZIP</layout>  
    </configuration>  
    <executions>  
    <execution>  
         <goals>  
             <goal>repackage</goal>  
         </goals>  
     </execution>  
   </executions>  
</plugin>  

以下是提取的修改项:

<configuration>
    <!--表示编译版本配置有效-->
    <fork>true</fork>
    <!--引入第三方jar包时,不添加则引入的第三方jar不会被打入jar包中-->
    <includeSystemScope>true</includeSystemScope>
    <!--排除第三方jar文件-->
    <includes>
        <include>
            <groupId>nothing</groupId>
            <artifactId>nothing</artifactId>
        </include>
    </includes>
</configuration>

修改的做用:

  • includeSystemScope:jar包分两种,一种是spring、mybatis等这种项目依赖的,再就是咱们外部手动引入的第三方 jar 依赖,若是该参数不设置为 true 的话是不能被打包进来的~
  • includes:这个节点就是排除项目中全部的 jar,那还怎么打包?

其实咱们须要将打包插件替换为 maven-jar-plugin,而后使用该插件拷贝依赖到 jar 到外面的 lib 目录。

二、maven-xxx-plugin

从 二、三、4 你会发现用到了 maven-xxx-plugin 格式的三种插件,简单说一下这三者的做用:

  • maven-compiler-plugin:

    完成对Java代码的编译,能够指定项目源码的jdk版本、编译后的jdk版本,以及编码,若是不写这个插件也是没问题的,不写会使用默认的 jdk 版原本处理,只是这样容易出现版本不匹配的问题,好比本地maven环境用的3.3.9版本,默认会使用jdk1.5进行编译,而项目中用的jdk1.8的编译环境,那就会致使打包时编译不经过。

  • maven-dependency-plugin:

    做用就是将全部依赖的jar文件复制到指定目录下,其中涉及到的 ${project.xx} 见下文补充。

  • maven-jar-plugin:

    主要做用就是将maven工程打包成jar包。

主要说一下 maven-jar-plugin 插件的以下配置:

<configuration>
    <!--资源文件不打进jar包中,作到配置跟项目分离的效果-->
    <excludes>
        <!-- 业务jar中过滤application.properties/yml文件,在jar包外控制 -->
        <exclude>*.properties</exclude>
        <exclude>*.xml</exclude>
        <exclude>*.yml</exclude>
    </excludes>
</configuration>

打包时排除资源配置文件,若是排除了配置文件那么项目启动是怎么读取呢?

配置文件有这么一个默认的优先级:

当前项目config目录下 > 当前项目根目录下 > 类路径config目录下 > 类路径根目录下

所以只须要将配置文件复制一份到与 jar 包平级目录下,或者与jar包平行config目录下,就能优先使用此配置文件,达到了伪分离目的。

最终的目录结构以下:

Maven 中的内置变量说明:

  • ${basedir} 项目根目录
  • ${project.build.directory} 构建目录,缺省为target
  • ${project.build.outputDirectory} 构建过程输出目录,缺省为target/classes
  • ${project.build.finalName} 产出物名称,缺省为{project.artifactId}-${project.version}
  • ${project.packaging} 打包类型,缺省为jar
  • ${project.packaging} 打包类型,缺省为jar
  • ${project.xxx} 当前pom文件的任意节点的内容

瘦身总结

Spring Boot 框架提供了一套本身的打包机制 — spring-boot-maven-plugin,Springboot 默认使用该插件来打包,打包时会将项目全部的依赖打入项目 jar 包里面,若是咱们想要抽离依赖的 jar 仅仅使用该插件是不行的,就须要将打包插件替换为 maven-jar-plugin,并拷贝全部的依赖到 jar 外面的 lib 目录。

项目打包时,在分离依赖 jar 包基础上,咱们又排除了配置文件,由于配置文件有一个默认的读取路径:

当前项目config目录下 > 当前项目根目录下 > 类路径config目录下 > 类路径根目录下

咱们只须要在当前项目 jar 包同级目录建立一个 config 文件夹,而后将配置文件复制一份,这样就达到了伪分离目的。

以后再修改配置文件,好比端口号、数据库链接信息等,就不须要从新打包项目了,直接修改完配置文件重启项目就能够了。

而通过分离依赖后的 jar 包从原来的100多兆到如今的1兆,若是后面须要变动业务逻辑,只须要轻量的编译项目,快速的实现项目的上传替换,有效的减小了网络开销,提升项目部署的效率。

博客地址:https://niceyoo.cnblogs.com

更多原创内容能够移步个人公众号,回复「面试」获取我整理的2020面经。

相关文章
相关标签/搜索