Spring 发布了 Spring Native 的 beta 版本,该功能已经在 start.spring.io 上可用了。
Spring Native 可以通过 GraalVM 将 Spring 应用程序编译成原生镜像,提供了一种新的方式来部署 Spring 应用。Spring Native 支持 Java 和 Kotlin。
这个项目的目标是寻找 Spring JVM 的替代方案,提供一个能将应用程序打包,并运行在轻量级容器的方案。期望能够在 Spring Native 中支持所有的 Spring 应用程序(几乎不用修改代码)。
GraalVM 介绍起来篇幅比较长,这里仅简要介绍。官网:https://www.graalvm.org/
GraalVM 是一个高性能的多语言运行时环境。设计目的是能够提高用 Java 和其他 JVM 语言编写的应用程序的执行速度,同时还为 JavaScript、Ruby、Python 和许多其他流行语言提供运行时。GraalVM 的多语言能力使得在一个应用程序中混合使用多种编程语言成为可能,同时消除了不同语言间互相调用的成本。详细内容可参考:Get Started with GraalVM
具体内容可参考:Lightweight cloud-native Java applications
构建 Spring Boot native 应用程序有 2 种方式:
本文只介绍第一种。
在待构建的机器上,必须安装了 Docker,可以参考 Get Docker,同时注意要能够以非 root 用户启动和运行。
可以通过使用 docker run hello-world
(不包含sudo
)命令检查 Docker daemon 是否可用。
一个简单的 Spring Boot Web 程序:
git clone https://github.com/spring-guides/gs-rest-service
cd gs-rest-service/complete
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.4.4</version>
<relativePath/>
</parent>
org.springframework.experimental:spring-native
提供了 native 配置的 API,例如 @NativeHint
这些 Spring 运行成 native image 的注解类。
<dependencies>
<!-- ... -->
<dependency>
<groupId>org.springframework.experimental</groupId>
<artifactId>spring-native</artifactId>
<version>0.9.1</version>
</dependency>
</dependencies>
Spring AOT 插件执行代码的提前转换,用以修复 native image 的兼容性。
<build>
<plugins>
<!-- ... -->
<plugin>
<groupId>org.springframework.experimental</groupId>
<artifactId>spring-aot-maven-plugin</artifactId>
<version>0.9.1</version>
<executions>
<execution>
<id>test-generate</id>
<goals>
<goal>test-generate</goal>
</goals>
</execution>
<execution>
<id>generate</id>
<goals>
<goal>generate</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
Spring Boot 的 Spring Boot Buildpacks support 可以将 Spring Boot 应用程序打包成一个容器。native image buildpack 可以通过 BP_NATIVE_IMAGE
环境变量开启。
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<image>
<builder>paketobuildpacks/builder:tiny</builder>
<env>
<BP_NATIVE_IMAGE>true</BP_NATIVE_IMAGE>
</env>
</image>
</configuration>
</plugin>
<repositories>
<!-- ... -->
<repository>
<id>spring-release</id>
<name>Spring release</name>
<url>https://repo.spring.io/release</url>
</repository>
</repositories>
<pluginRepositories>
<!-- ... -->
<pluginRepository>
<id>spring-release</id>
<name>Spring release</name>
<url>https://repo.spring.io/release</url>
</pluginRepository>
</pluginRepositories>
mvn spring-boot:build-image
通过此命令,可以创建一个使用 GraalVM native image compiler 构建的 Linux 容器,默认情况下,这个镜像是在本地。
可以使用 docker images 命令,查看镜像:
使用 docker 启动这个镜像:
docker run --rm -p 8080:8080 rest-service:0.0.1-SNAPSHOT
本次启动时间是47ms
,而 JVM 的程序启动一般都是1500ms
左右。
现在服务已经启动了,可以通过 localhost:8080/greeting 访问服务。在浏览器中可以看到:
{"id":1,"content":"Hello, World!"}
若编译时遇到下面的情况,则表明构建时没有 docker 权限,如果配置一直不成功,可以直接在 mvn spring-boot:build-image
命令前加个 sudo
。
若编译时遇到下面的情况,是 OOM 问题,需要把 Docker 的内存改大(8G)。
以 Mac 的 Docker Client 设置为例:
这部分参考自:云原生时代,Java 的危与机
JVM 的程序运行时间长,是因为存在虚拟机的初始化和类加载过程,如果将字节码直接编译成原生代码,则可以彻底解决这些问题。同时因为没有即时编译器在运行时编译,所有代码都在编译期编译和优化。因为少了 Java 虚拟机、即时编译器这些额外组件,原生程序也能够省去它们原本消耗的内存资源和镜像体积。
Java 支持提前编译最大的困难,在于 Java 是一门动态链接的语言,它假设程序的代码空间是开发的,允许在程序的任何时候通过类加载器去加载新的类,作为程序的一部分。要进行提前编译,就必须放弃这部分动态性,所有要运行的代码必须在编译期全部可知。这样动态加载、反射(通过反射可以调用在编译期不可知的方法)、动态代理、字节码生成库(如 CGLib)等一切会运行时产生新代码的功能都不再可用。
当然 Spring Native 遇到的问题有很多,且仍然处于试验阶段。以原生方式运行后,启动时间是能够缩短很多,但是程序的运行效率还是若于传统基于 JVM 的方式,且编译成原生程序代码的时间更长。
项目 GitHub LeetCode 全解,欢迎大家 star、fork、merge,共同打造最全 LeetCode 题解!
Java 编程思想-最全思维导图-GitHub 下载链接,需要的小伙伴可以自取~!!!
Spring Native 项目,把 Spring 项目编译成原生程序!
原文:https://www.cnblogs.com/510602159-Yano/p/14591079.html