GraalVM?native-image編譯后quarkus的超音速啟動(dòng)
前言
quarkus號(hào)稱超音速亞原子JAVA為Graalvm量身定制的java堆棧,是否名副其實(shí)呢?下面就來(lái)看看真實(shí)情況如何。動(dòng)手前先簡(jiǎn)單介紹下Graalvm,它是oracle出品的一個(gè)AOT編譯器,可以將應(yīng)用程序編譯成本地映像,通俗的說(shuō)可以將java編譯成機(jī)器可直接執(zhí)行的程序,可以參考go語(yǔ)言的編譯輸出產(chǎn)物。而且graalvm不僅僅支持java,對(duì)其他語(yǔ)言也有很好的支持。下面先看一張quarkus的java應(yīng)用程序在傳統(tǒng)的vm下面和graalvm下面的資源占用圖。
graalvm:https://www.graalvm.org/
native-image編譯配置
<profiles> <profile> <id>native</id> <activation> <property> <name>native</name> </property> </activation> <build> <plugins> <plugin> <groupId>io.quarkus</groupId> <artifactId>quarkus-maven-plugin</artifactId> <version>${quarkus-plugin.version}</version> <executions> <execution> <goals> <goal>native-image</goal> </goals> <configuration> <enableHttpUrlHandler>true</enableHttpUrlHandler> <reportErrorsAtRuntime>true</reportErrorsAtRuntime> </configuration> </execution> </executions> </plugin> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-failsafe-plugin</artifactId> <version>${surefire-plugin.version}</version> <executions> <execution> <goals> <goal>integration-test</goal> <goal>verify</goal> </goals> <configuration> <systemPropertyVariables> <native.image.path>${project.build.directory}/${project.build.finalName}-runner</native.image.path> <java.util.logging.manager>org.jboss.logmanager.LogManager</java.util.logging.manager> <maven.home>${maven.home}</maven.home> </systemPropertyVariables> </configuration> </execution> </executions> </plugin> </plugins> </build> </profile> </profiles>
在pom文件中使用profile新增了一個(gè)native編譯環(huán)境,引入了quarkus的編譯插件,并激活了native的編譯。實(shí)際上,這個(gè)插件只會(huì)幫你將graalvm編譯指令編排好,graalvm的環(huán)境還需要你自己搭建,quarkus每個(gè)迭代的版本會(huì)針對(duì)特定的graalvm版本做優(yōu)化,所以不是所有的版本都相互兼容的。比如quarkus1.5.final版本兼容graalvm19.x版本,最新的quarkus1.6.final支持graalvm20.1.1版本,各版本下載地址,下載下來(lái)后,和配置java環(huán)境一樣,將目錄添加到GRAALVM_HOME環(huán)境變量中即可,如:
最終quarkus的maven編譯插件會(huì)幫我們生成一條這樣的graalvm編譯指令,如:
F:\runtime\graalvm-ce-java8-19.3.1\bin\native-image.cmd -J-Dsun.nio.ch.maxUpdateArraySize=100 -J-Djava.util.logging.manager=org.jboss.logmanager.LogManager -J-DCoordinatorEnvironmentBean.transactionStatusManagerEnable=false -J-Dvertx.logger-delegate-factory-class-name=io.quarkus.vertx.core.runtime.VertxLogDelegateFactory -J-Dvertx.disableDnsResolver=true -J-Dio.netty.leakDetection.level=DISABLED -J-Dio.netty.allocator.maxOrder=1 -J-Duser.language=zh -J-Dfile.encoding=UTF-8 --initialize-at-run-time=java.net.Inet4Address -H:+TraceClassInitialization --initialize-at-build-time= -H:InitialCollectionPolicy=com.oracle.svm.core.genscavenge.CollectionPolicy$BySpaceAndTime -H:+JNI -jar kk-org-thansfer-admin-1.0-SNAPSHOT-runner.jar -H:FallbackThreshold=0 -H:+ReportUnsupportedElementsAtRuntime -H:+ReportExceptionStackTraces -H:+AddAllCharsets -H:+IncludeAllTimeZones -H:EnableURLProtocols=http,https --enable-all-security-services -H:-UseServiceLoaderFeature -H:+StackTrace kk-org-thansfer-admin-1.0-SNAPSHOT-runner
以上就是quarkus集成graalvm編譯環(huán)境的所有內(nèi)容了,但是graalvm在windows系統(tǒng)下的編譯并不友好,博主嘗試過(guò)很多方法,包括通過(guò)docker容器掛載編譯,都宣告失敗了,所以如果你也有同樣的問(wèn)題,看下我們的異常是否一樣:
[ERROR] Failed to execute goal io.quarkus:quarkus-maven-plugin:1.6.0.Final:native-image (default) on project kk-org-thansfer-admin: Failed to generate native image: io.quarkus.builder.BuildException: Build failure: Build failed due to errors [ERROR] [error]: Build step io.quarkus.deployment.pkg.steps.NativeImageBuildStep#build threw an exception: java.lang.RuntimeException: Failed to build native image [ERROR] at io.quarkus.deployment.pkg.steps.NativeImageBuildStep.build(NativeImageBuildStep.java:366) [ERROR] at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) [ERROR] at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) [ERROR] at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) [ERROR] at java.lang.reflect.Method.invoke(Method.java:498) [ERROR] at io.quarkus.deployment.ExtensionLoader$2.execute(ExtensionLoader.java:932) [ERROR] at io.quarkus.builder.BuildContext.run(BuildContext.java:277) [ERROR] at org.jboss.threads.ContextClassLoaderSavingRunnable.run(ContextClassLoaderSavingRunnable.java:35) [ERROR] at org.jboss.threads.EnhancedQueueExecutor.safeRun(EnhancedQueueExecutor.java:2046) [ERROR] at org.jboss.threads.EnhancedQueueExecutor$ThreadBody.doRunTask(EnhancedQueueExecutor.java:1578) [ERROR] at org.jboss.threads.EnhancedQueueExecutor$ThreadBody.run(EnhancedQueueExecutor.java:1452) [ERROR] at java.lang.Thread.run(Thread.java:745) [ERROR] at org.jboss.threads.JBossThread.run(JBossThread.java:479) [ERROR] Caused by: java.lang.RuntimeException: Image generation failed. Exit code: -1073741819 [ERROR] at io.quarkus.deployment.pkg.steps.NativeImageBuildStep.imageGenerationFailed(NativeImageBuildStep.java:416) [ERROR] at io.quarkus.deployment.pkg.steps.NativeImageBuildStep.build(NativeImageBuildStep.java:344) [ERROR] ... 12 more [ERROR] -> [Help 1]
不過(guò)別慌,博主還沒(méi)放棄,下面通過(guò)docker多段鏡像編排解決問(wèn)題,上面貼的pom配置代碼別刪
docker多段鏡像編排
## Stage 1 : build with maven builder image with native capabilities FROM quay.io/quarkus/centos-quarkus-maven:20.1.0-java8 AS build COPY pom.xml /usr/src/app/ COPY src /usr/src/app/src USER root RUN chown -R quarkus /usr/src/app USER quarkus RUN mvn -f /usr/src/app/pom.xml -Pnative clean install -Dmaven.test.skip=true -Denv=DEV ## Stage 2 : create the docker final image FROM registry.access.redhat.com/ubi8/ubi-minimal WORKDIR /work/ COPY --from=build /usr/src/app/target/*-runner /work/application # set up permissions for user `1001` RUN chmod 775 /work /work/application \ && chown -R 1001 /work \ && chmod -R "g+rwX" /work \ && chown -R 1001:root /work EXPOSE 8080 USER 1001 CMD ["./application", "-Dquarkus.http.host=0.0.0.0"]
第一階段,基于quarkus的centos基礎(chǔ)鏡像,里面內(nèi)置了graalvm環(huán)境,然后我們只需要將代碼和pom配置copy進(jìn)系統(tǒng)里,同鏡像里的環(huán)境編譯成native-image,然后第二段,基于小紅帽的基礎(chǔ)鏡像運(yùn)行環(huán)境,將構(gòu)建的產(chǎn)物copy進(jìn)去,如此即完成了docker容器的構(gòu)建。不過(guò)這樣的方式構(gòu)建,所有的依賴都是即時(shí)下載的,對(duì)本地網(wǎng)絡(luò)要求會(huì)比較高,整體編譯時(shí)長(zhǎng)會(huì)比較長(zhǎng)。網(wǎng)絡(luò)稍微一抖動(dòng)就會(huì)編譯失敗,所以最好在pom里配置下國(guó)內(nèi)比較快的maven倉(cāng)庫(kù),比如阿里云的maven倉(cāng)庫(kù)。還有,在graalvm編譯階段,會(huì)非常的吃內(nèi)存,這個(gè)時(shí)候它會(huì)加載所有的代碼用于靜態(tài)分析,這塊內(nèi)容阿里巴巴的jvm團(tuán)隊(duì)有做過(guò)優(yōu)化,后面可能會(huì)來(lái)做一個(gè)分享。
可能遇到的問(wèn)題,graalvm是在編譯時(shí)初始化的,所有有些依賴如果只能運(yùn)行時(shí)初始化,可以在quarkus中添加如下的配置:
quarkus.native.additional-build-args=--initialize-at-run-time=java.net.Inet4Address
效果展示
docker編譯雖然會(huì)比較慢,但是最后還是成功了,下面展示下quarkus的神奇之處,當(dāng)鏡像成功運(yùn)行起來(lái)那一刻,博主還是按捺不住心中的喜悅之情,他么的跟中了500W似的,注意,博主的這個(gè)程序不是簡(jiǎn)單的hello,而是有數(shù)據(jù)源又接口的生產(chǎn)級(jí)CURD的程序。
native-image啟動(dòng)時(shí)間
jvm下的啟動(dòng)時(shí)間
除了啟動(dòng)時(shí)間提升了n倍之外,內(nèi)存占用也是非常感人,native-image在容器里面總內(nèi)存占用才90M,而在jvm下面應(yīng)用的內(nèi)存占用就要300M左右了。即使如此,相比于spring boot的動(dòng)輒1G的內(nèi)存占用,也已經(jīng)表現(xiàn)的十分的優(yōu)秀了。
結(jié)語(yǔ)
當(dāng)應(yīng)用啟動(dòng)起來(lái)才1s不到的時(shí)候,博主是發(fā)自內(nèi)心的高興呀,quarkus的超音速亞原子是名副其實(shí)的。雖然目前graalvm還有諸多的問(wèn)題,比如編譯環(huán)境兼容性,對(duì)第三方依賴的兼容性,博主引入的dubbo就會(huì)有問(wèn)題,最后只能排除,不過(guò)quarkus體系的依賴都是經(jīng)過(guò)優(yōu)化的,可以放心的使用。不過(guò)不用擔(dān)心,我相信graalvm技術(shù)是java主力探索方向,是未來(lái)。而且有阿里巴巴這種級(jí)別的jvm團(tuán)隊(duì)一起在優(yōu)化,graalvm會(huì)越來(lái)越成熟。博主已經(jīng)迫不及待的要在下一個(gè)項(xiàng)目用起來(lái)了,有任何quarkus和graalvm兩個(gè)的問(wèn)題都可以找我哦
以上就是GraalVM native-image編譯后quarkus的超音速啟動(dòng)的詳細(xì)內(nèi)容,更多關(guān)于GraalVM native-image編譯后quarkus的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot在自定義類中調(diào)用service層等Spring其他層操作
這篇文章主要介紹了SpringBoot在自定義類中調(diào)用service層等Spring其他層操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06Netty網(wǎng)絡(luò)編程零基礎(chǔ)入門(mén)
Netty是一個(gè)異步的、基于事件驅(qū)動(dòng)的網(wǎng)絡(luò)應(yīng)用框架,用于快速開(kāi)發(fā)可維護(hù)、高性能的網(wǎng)絡(luò)服務(wù)器和客戶端,如果你還不了解它的使用,就趕快繼續(xù)往下看吧2022-08-08詳解在Spring中如何自動(dòng)創(chuàng)建代理
這篇文章主要介紹了詳解在Spring中如何自動(dòng)創(chuàng)建代理,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07SpringBoot自定義注解及AOP的開(kāi)發(fā)和使用詳解
在公司項(xiàng)目中,如果需要做一些公共的功能,如日志等,最好的方式是使用自定義注解,自定義注解可以實(shí)現(xiàn)我們對(duì)想要添加日志的方法上添加,這篇文章基于日志功能來(lái)講講自定義注解應(yīng)該如何開(kāi)發(fā)和使用,需要的朋友可以參考下2023-08-08詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí)
本篇文章主要介紹了詳解Spring Cloud Hystrix斷路器實(shí)現(xiàn)容錯(cuò)和降級(jí),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05MyBatis Plus關(guān)閉SQL日志打印的方法
這篇文章主要介紹了MyBatis-Plus如何關(guān)閉SQL日志打印,文中通過(guò)圖文結(jié)合講解的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2024-02-02Java實(shí)現(xiàn)一個(gè)簡(jiǎn)單的長(zhǎng)輪詢的示例代碼
長(zhǎng)輪詢是與服務(wù)器保持即時(shí)通信的最簡(jiǎn)單的方式,它不使用任何特定的協(xié)議,例如 WebSocket ,所以也不依賴于瀏覽器版本等外部條件的兼容性。本文將用Java實(shí)現(xiàn)一個(gè)簡(jiǎn)單的長(zhǎng)輪詢,需要的可以參考一下2022-08-08Java即將引入新對(duì)象類型來(lái)解決內(nèi)存使用問(wèn)題
這篇文章主要介紹了Java即將引入新對(duì)象類型來(lái)解決內(nèi)存使用問(wèn)題,文章通過(guò)圍繞主題的相關(guān)資料展開(kāi)詳細(xì)內(nèi)容,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-05-05