Redian新闻
>
pandora boot热点应用探索60秒构建之路

pandora boot热点应用探索60秒构建之路

科技


背景


在阿里内部有一个典型的pandora boot应用A,它参与研发的同学与每周的发布次数都比较多,是一个典型的热点应用.它的构建产物是一个fatjar,文件大小有1个G,其中包含了2893个jar。
在最近,应用A使用了最新的 amaven 版本后,p95的构建耗时从20分钟下降到了6分钟。
我们来看看有没可能将应用A的构建继续优化,下降到60秒.这样,我们就有更多的时间去外面呼吸更多的新鲜空气了。

现状


在命中所有amaven的依赖树缓存,及docker缓存后(即最佳情况),我们拿应用A的master分支来构建下,看能到几分钟。
一次典型的构建主要分二个大步骤,一个是主包构建,即打fatjar,一个是镜像构建。
我们先看主包构建,在amaven的依赖缓存全命中的情况下,最优是2:29min。
再看镜像构建,build耗时28秒,push耗时38秒,共66秒。
所以,在最佳情况下,应用A的"纯构建"耗时在3分半,即210秒左右.但实际上左边菜单中显示的时间169+73=242秒,约有242-210=32秒的时间,是构建系统在做构建前置及后置任务。

方案

3.1 mvn build优化

从amaven的build report中可以看到top 3慢的步骤,耗时依次是34390ms,20376ms,15761ms;主要是慢在maven-compiler-plugin及autoconfig-maven-plugin这二个插件,我们对这二个插件分别来优化。

3.1.1 增量编译 最佳减少45秒

maven-compiler-plugin主要是执行以下命令:
javac -classpath a.jar:b.jar.... --source-path ..

这个插件会将所有maven根据应用的pom进行依赖分析并且仲裁后的jar包列表作为classpath参数.maven-compiler-plugin的耗时基本上主是javac的耗时,而javac的耗时长并不是因为要编译的java文件太多,而更是因为classpath中的jar包太多;当它编译一个java类时,对这java类中import进来的类要去这些classpath指定的jar包中去遍历查找.从文章开头就讲到的应用A最后的fatjar中有2893个jar,可以近似知道classpath中的jar包也不少。
所以减少classpath中的jar包数,即通过应用的pom来治理依赖与减少依赖,是一个更治本的方案.但我们今天要说的是,不执行javac肯定是最快的。
不执行javac,而是直接复用之前编译的class文件,是最快的.即增量编译.即只编译变化的java,没变的就直接复用.所以增量编译,其实是少执行javac,而并不全是不执行javac。
现在amaven实现的增量编译,不是基于单个java文件,而是一个maven module.一般应用典型的module依赖关系如下图:
biz中的代码会高频更新,但它依赖的dal,common相对稳定.当一个分支在第二次编译时,只修改了biz层,则只要重新编译biz,controller二层,而dal,common二层的class可以直接复用之前的. 增量编译的一些描述问题请点这 并搜索"增量".在这我们先来看看增量编译的效果。
从上图可以看出 从原来02:20min能降到01:35min即减少45秒左右。
那如何启用增量编译?
只要在使用amaven进行编译时,加上参数-DenableIncrementTask=true 。

3.1.2 autoconfig 固定减少30秒

"增量编译"有效果,但其实是不稳定的,要看一个应用每次编译时是否只修改了java类,且这java类是否在上层模块.现在我们再来看看autoconfig插件.对这插件的优化的效果是稳定的。
autoconfig插件的作用是将同一份代码用不同的配置项来编译,从而部署在不同环境。
在之前我们已经对autoconfig针对war包应用的场景作过一次优化.现在我们再来对fatjar应用的场景作优化。
从优化前的日志中可以看到二点:1.是日志中有allocating large array,即在执行过程中消耗了大量的内存,因为当autoconfig插件执行时是会将一个应用A约1G大小的fatjar以zipInputStream的方式读进内存,再以nextEntry的方式遍历所有文件; 2.是耗时了34秒。
目前的一个fatjar应用主包的形态(目录或jar包文件)的构建与部署过程一般如下:
即mvn build时先构建出目录,然后再压缩成jar;到最后应用启动时又会将jar包解压成目录.压缩成jar主要目的是减少体积,但却带来了CPU开销.在网络带宽资源大于CPU资源时我们推荐不要压缩成jar包。
直接是目录形态.它还有2个好处:
  1. autoconfig可以并发执行;
  2. docker build可以使用SYNC语法;

请参照以下步骤来升级autoconfig:
1、在构建配置文件中配置以下二个参数:其中第一个是让aone不要压缩成tgz,而第二个是让aone知道要将哪个目录copy到镜像中。
build.output.copyonly=true build.output=appA-bootstrap-start/target/appA

2、将autoconfig-plugin 放到maven-antrun-plugin后,且使用2.0.10及以上版本,再加上以下第24-28行的配置.其中第25行,要指定到应用的fatjar结构的目录中的lib目录,即jar包们所在的目录。
<plugin><artifactId>maven-antrun-plugin</artifactId><executions>    <execution>        <phase>package</phase>        <configuration>            <tasks>                <unzip                        src="${project.build.directory}/${project.build.finalName}.${project.packaging}"                        dest="${project.build.directory}/appA"/>            </tasks>        </configuration>        <goals>            <goal>run</goal>        </goals>    </execution></executions></plugin>
<plugin><groupId>com.alibaba.citrus.tool</groupId><artifactId>autoconfig-plugin</artifactId><version>2.0.10</version><configuration> <dest>${project.build.directory}/appA/BOOT-INF/lib</dest> <type>jar</type> <isListDestFiles>true</isListDestFiles></configuration><executions> <execution> <phase>package</phase> <goals> <goal>autoconfig</goal> </goals> </execution></executions></plugin>

maven-antrun-plugin会将fatjar包文件解压成目录,所以要放在autoconfig插件前. 那能否让pandora-boot-maven-plugin不将fatjar压缩成文件呢?因为压缩的时间开销还好,经过对appA的测试,将2893个jar压缩成一个fat.jar只要2秒多(因为pandora-boot-maven-plugin没使用二次压缩,所以很快).同时因为要修改这插件不压缩成fat.jar,修改地方较多,所以pandora-boot-maven-plugin暂不提供 不压缩成fat.jar 的版本。
3、修改dockerfile:
COPY ${APP_NAME}.tgz /home/admin/${APP_NAME}/target/${APP_NAME}.tgz

修改成:
COPY build-output/ /home/admin/${APP_NAME}/target/${APP_NAME}/

因为现在不是tgz,而是build-output目录了.
4、修改应用启动脚本
因为现在不是fat.jar一个文件,而是一个目录了.所以对于旧版本的pandora boot应用,要相应修改应用的启动脚本,如原来有解压操作的,现在可以不要了.而最新版本的pandora boot版本的应用,则兼容,不用修改。
最后,这样配置后的效果如下:
从上图可以看出autoconfig的耗时从34秒下降到了3.8s,即减少了30秒。
autoconfig插件升级后,我们通过对比优化前后的二个构建日志来验证结果的正确性.搜索日志中的autoconfig产生的 "Generating META-INF"数量是一样的.比较具体的配置了的jar包列表,也是相同的,说明结果一样。
同时在 优化后日志 中我们发现:Runtime : ran out of parsers. 的日志:
这些日志是velocity报的,因为现在是"多线程作配置",所以同时要有的parser会较多。
对于新版本的autoconfig插件,我们主要作了二个优化:
1.用线程池来并发执行config:
2.能并发执行的前提是要autoconfig的目标是一个目录,而不是一个fat.jar文件.当目标是目录时,会先listFile,再将fileList传给destFiles。
在"增量编译"与autoconfig"并发执行"二个优化后,最佳情况下的mvn构建耗时能到55秒左右。

3.2 docker build优化

应用A的dockerfile 片段如下:
首先,是COPY主包没在最后一行,导致第14行每次编译都会执行,因为主包每次构建都会变。
从日志来看,13-14二行,执行了28-21=7秒左右. 即如能根据dockerfile的最佳实践"将不变的放下层,变化的放上层",将13,14二行放到10行前,可以节省7秒左右。
接着,因为前面我们将主包从tgz变成了build-output目录了,所以还可以使用 SYNC语法。
只要一步操作即可:修改dockerfile,将COPY修改成SYNC。
即将
COPY build-output/ /home/admin/${APP_NAME}/target/${APP_NAME}/

修改成(注意,SYNC不支持PATH中变量,请换成具体的应用名;最后也不能有/):
SYNC build-output/ /home/admin/appA/target/appA

最后,我们看看效果。
使用SYNC后,从原来 一个fatjar要1G的内容要docker build与push,变成只有变化的jar包(源码产生的及要autoconfig的,约100个jar包)才要增量build.耗时能从66秒减少30秒左右。

可行性小结

最后综合三个优化点后来看,一次完整的构建只能从242s降到136s,离60s还有一段路要突破。
但打个折,只从mvn 构建来看,只要进行autoconfig插件升级与启用增量构建,就可以到达60s(纯mvn build可以达到44秒)。
所以让我们行动起来:
  1. 启用amaven增量编译;

  2. 升级autoconfig插件;

  3. 选择buildkit编译机并使用SYNC;
对于60s的追求,我们也会一直探索下去,即使又到来年的丹桂飘香时......
amaven与buildkit相关内容请参考:java应用提速(速度与激情)

往期推荐

1.深入理解virtual关键字

2.《领域驱动设计》:从领域视角深入仓储(Repository)的设计和实现

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
纽约市皇后区长岛市 这里只有condo没有coop 买房和租房需知"浮盈"变"实盈"!基金11月分红很"壕",环比暴增100%,这些产品最给力US Treasury Series I Savings Bonds (I Bonds) 简介::11月利率将变为6.48%Intel i5-4460 CPU + MSI H81M-P33 motherboard bundle万和证券斩获"新锐投行君鼎奖",探索"精品投行"特色路平价买到高级感!IKEA全新 OBEGRÄNSAD系列,全系列都好看!哇!Torvalds 为自己的 Apple M2 Macbook 专门修改了 Fedora Linux | Linux 中国怎么发热点弯道超车?这33篇顶刊综述带你了解6大热点的前世今生!At Ramsar COP14, China Pledges to Expand and Protect WetlandsFedora Workstation引入原生支持Wayland 的 Blender读懂这30+热点综述,师兄抓住热点发2篇SCI!性能爆表:SpringBoot利用ThreadPoolTaskExecutor批量插入百万级数据实测!【祝贺美景上任】《大鱼》这是由15个单摆组成的,整个周期约为60秒,60秒后,一切恢复原状再跳一次!IKEA x OBEGRÄNSAD联名!宜家22年最受瞩目系列开售!北美求职60秒:前端工程师(Front-End Engineer)需要掌握哪些技能?(11月第2周)美国女贼赌城偷名表塞入YD 警察搜身无果动用探测器Argentina won World Cup 2022——阿根廷!"Don't go down" 梅西圆梦五四运动,顾维钧,很有意思,巴黎和谈的英雄,民国的全权大使,共和国的战犯,国际法院的院长,,,张铁生改变了我的人生坚持自力更生独立自主崛起事实Good Book with Red Book,假期我们一起 Hard Work!“我们的祖先到底是谁?为何智人胜出?”丨2022诺奖深入回答了这些问题。附Svante Pääbo趣闻古人类DNA与重症新冠有关?2022诺奖得主Pääbo,竟是前诺奖得主私生子hǎo xiǎng “rua” 🤩China Bans Celebs From Endorsing Tutoring, Tobacco, and Others疫中球,诗二首绝美长岛市!这里只有condo没有coop,买房、租房有哪些需要知道的?专家详解构建下一代万亿级云原生消息架构:Apache Pulsar 在 vivo 的探索与实践Windows Terminal正式成为Windows 11默认终端Chase Marriott Bonvoy Boundless 信用卡【100k or 75k+$300 开卡奖励】SpringBoot 整合 Groovy 脚本,实现动态编程
logo
联系我们隐私协议©2025 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。