vlambda博客
学习文章列表

搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+

大家好,我是D哥

搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+
Java面试那些事儿
回复关键字 java ,领取大厂最新面试题、简历模板和架构师路线图。本号分享内容主要包括Java基础、Spring Cloud、Intellij IDEA、Dubbo、Zookeeper、Redis、架构设计、微服务、架构师等。
274篇原创内容
Official Account
作者:集成显卡
来源:https://blog.csdn.net/ssrc0604hx/article/details/54175027

# 背景


随着Spring Boot的流行,越来越多开发者选择使用Spring Boot来发布Web应用。不同于传统的War包发布,Spring Boot把整个项目打包成一个可运行的Jar包(即所谓的Flat Jar),导致了这个Jar包很大(通常有40M+)。如今迭代发布时常有的事情,每次都上传一个如此庞大的文件,会浪费很多时间。


下面就以一个小项目为例,简述小弟所用的瘦身方案。当然如果是内网发布或者你用的宽带异常给力,瘦身就没有多大意义了。


# 实践


项目简介


新建一个练习用的项目,其结构如下


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+


  1. ht-cdn存放的是静态资源(如第三方js、css、images等)

  2. ht-domain项目中的数据实体定义

  3. ht-repository数据层接口及实现

  4. ht-service业务逻辑接口及实现

  5. ht-ui-webWeb管理


其中ht-ui-web依赖于ht-domain、ht-repository、ht-service,实现了一个简单的GetMapping。


接着打包项目,整个jar包24M这样


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+


# 瘦身准备


首先我们要对Jar包有一个初步认识,它的内部结构如下

example.jar | +-META-INF | +-MANIFEST.MF +-org | +-springframework | +-boot | +-loader | +-<spring boot loader classes> +-BOOT-INF +-classes | +-mycompany | +-project | +-YourClasses.class +-lib +-dependency1.jar +-dependency2.jar


运行该Jar时默认从BOOT-INF/classes加载class,从BOOT-INF/lib加载所依赖的Jar包。如果想要加入外部的依赖Jar,可以通过设置环境变量LOADER_PATH来实现。


如此一来,就可以确认我们的思路了:


1. 把那些不变的依赖Jar包(比如spring依赖、数据库Driver等,这些在不升级版本的情况下是不会更新的)从Flat Jar中抽离到单独的目录,如libs


2. 在启动Jar时,设置LOADER_PATH使用上一步的libs


这样,我们最终打包的jar包体积就大大减少,每次迭代后只需要更新这个精简版的Jar即可。

# 具体步骤


打包时瘦身


通常我们是用spring-boot-maven-plugin进行打包,通过阅读文档发现可以通过配置使得该插件在打包时忽略特定的依赖,文档在spring-boot-maven-plugin。


首先备份一下原先的依赖:编译打包成Flat Jar,然后将BOOT-INF/lib下的除去ht-*相关的jar文件全部解压出来,另存到libs目录下。


接着修改pom.xml配置如下

<build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <layout>ZIP</layout> <!--去除在生产环境中不变的依赖--> <excludeGroupIds> org.springframework.boot, org.springframework, org.springframework.data, org.mongodb, com.github.0604hx, com.fasterxml.jackson.core, commons-beanutils, commons-codec, org.apache.commons, org.apache.tomcat.embed, org.hibernate, org.slf4j, com.jayway, org.jboss, com.alibaba, com.fasterxml, commons-collections, ch.qos.logback, org.scala-lang, org.yaml, org.jboss.logging, javax.validation, nz.net.ultraq.thymeleaf, org.thymeleaf, ognl, org.unbescape, org.javassist </excludeGroupIds> </configuration> </plugin> </plugins></build>


此时打包好的ht-ui-web.jar只有117kb这样


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+


BOOT-INF/lib下只有ht相关的jar


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+


但是由于没有其他依赖,ht-ui-web.jar是不能如期运行起来的

java -jar ht-ui-web-1.0.jarException in thread "main" java.lang.reflect.InvocationTargetException at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at org.springframework.boot.loader.MainMethodRunner.run(MainMethodRunner.java:48) at org.springframework.boot.loader.Launcher.launch(Launcher.java:87) at org.springframework.boot.loader.Launcher.launch(Launcher.java:50) at org.springframework.boot.loader.PropertiesLauncher.main(PropertiesLauncher.java:521)Caused by: java.lang.NoClassDefFoundError: org/springframework/boot/SpringApplication at com.nerve.huotong.web.WebApplication.main(WebApplication.java:21) ... 8 moreCaused by: java.lang.ClassNotFoundException: org.springframework.boot.SpringApplication at java.net.URLClassLoader.findClass(Unknown Source) at java.lang.ClassLoader.loadClass(Unknown Source) at org.springframework.boot.loader.LaunchedURLClassLoader.loadClass(LaunchedURLClassLoader.java:94) at java.lang.ClassLoader.loadClass(Unknown Source) ... 9 more


至此我们要设置LOADER_PATH,如下

java -Dloader.path="libs/" -jar ht-ui-web.jar


便可以看到熟悉的Spring Boot 启动信息了。


# 继续瘦身


上面的项目结构介绍提到了ht-cdn,我是把前端用到的库都放在这里。然后单独启动一个Web Application。其他项目需要用到静态资源就直接使用。


还有另外一个做法是,把resources/public直接丢到libs下(就是跟上一步剥离出来的jar包放在一起),结构如下:


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+


这样也是可以的(不过要注意不能跟真实项目中自己写的静态资源重名)。


# 结语


经过上面的瘦身,每次迭代开发开的Jar包就显得苗条多了。


插些题外话


Spring Boot 中的 banner.txt


banner是spring boot 应用启动时打印在控制台的彩蛋信息,默认是这样的

 . ____ _ __ _ _ /\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \ \\/ ___)| |_)| | | | | || (_| | ) ) ) ) ' |____| .__|_| |_|_| |_\__, | / / / / =========|_|==============|___/=/_/_/_/ :: Spring Boot :: (v1.4.3.RELEASE)


想要修改这个文本的话,只需要在resources下新建banner.txt即可。


这里可以自定义banner:http://patorjk.com/software/taag



 
   
   
 
搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+技术交流群搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+
  
    
    
  


最后,D哥也建了一个技术群,主要探讨一些新的技术和开源项目值不值得去研究及IDEA使用的“骚操作”,有兴趣入群的同学,可长按扫描下方二维码,一定要备注:城市+昵称+技术方向,根据格式备注,可更快被通过且邀请进群。


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+

▲长按扫描


搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+热门推荐:

   
     
     
   




搞了一个迭代发布下SpringBoot Jar瘦身方案,老大给我打了个A+