Spring Native 项目,把 Spring 项目编译成原生程序!
Spring 发布了 Spring Native 的 beta 版本,该功能已经在 start.spring.io 上可用了。
Spring Native 是什么
Spring Native 可以通过 GraalVM 将 Spring 应用程序编译成原生镜像,提供了一种新的方式来部署 Spring 应用。Spring Native 支持 Java 和 Kotlin。
这个项目的目标是寻找 Spring JVM 的替代方案,提供一个能将应用程序打包,并运行在轻量级容器的方案。期望能够在 Spring Native 中支持所有的 Spring 应用程序(几乎不用修改代码)。
优点
- 编译出来的原生 Spring 应用可以作为一个独立的可执行文件进行部署(不需要安装 JVM)
- 几乎瞬时的启动(一般小于 100 毫秒)
- 瞬时的峰值性能
- 更低的资源消耗
缺点
- 比 JVM 更长的构建时间
- 相比于传统的 Java 运行方式,运行时优化不足
原生镜像(native image)和常规 JVM 程序的区别
- 在构建时会从主入口点,静态分析应用程序
- 在构建时会移除未使用的代码
- 需要配置反射、动态代理等
- classpath 在构建时就已经确定
- 没有类延迟加载:可执行文件中所有的内容都会在启动时加载到内存中
- 在构建时就运行了一些代码
- 构建原生镜像还存在一些 局限性
前置条件:GraalVM
GraalVM 介绍起来篇幅比较长,这里仅简要介绍。官网:https://www.graalvm.org/
GraalVM 是一个高性能的多语言运行时环境。设计目的是能够提高用 Java 和其他 JVM 语言编写的应用程序的执行速度,同时还为 JavaScript、Ruby、Python 和许多其他流行语言提供运行时。GraalVM 的多语言能力使得在一个应用程序中混合使用多种编程语言成为可能,同时消除了不同语言间互相调用的成本。详细内容可参考:Get Started with GraalVM
支持的语言
关键特性
GraalVM 下的 Java 微服务
具体内容可参考:Lightweight cloud-native Java applications
Spring Native 的 Hello World
构建 Spring Boot native 应用程序有 2 种方式:
- 使用 Spring Boot Buildpacks support 构建一个包含本地可执行文件的轻量级容器。
- 使用 the GraalVM native image Maven plugin support 构建一个本地可执行文件。
本文只介绍第一种。
系统要求
在待构建的机器上,必须安装了 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
配置 Spring Boot 版本
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.4.4</version>
<relativePath/>
</parent>
添加 Spring Native 依赖
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 插件
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>
开启 native image 支持
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>
Maven Repository
<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 设置为例:
Spring Native 所遇到的问题
这部分参考自:云原生时代,Java 的危与机
JVM 的程序运行时间长,是因为存在虚拟机的初始化和类加载过程,如果将字节码直接编译成原生代码,则可以彻底解决这些问题。同时因为没有即时编译器在运行时编译,所有代码都在编译期编译和优化。因为少了 Java 虚拟机、即时编译器这些额外组件,原生程序也能够省去它们原本消耗的内存资源和镜像体积。
Java 支持提前编译最大的困难,在于 Java 是一门动态链接的语言,它假设程序的代码空间是开发的,允许在程序的任何时候通过类加载器去加载新的类,作为程序的一部分。要进行提前编译,就必须放弃这部分动态性,所有要运行的代码必须在编译期全部可知。这样动态加载、反射(通过反射可以调用在编译期不可知的方法)、动态代理、字节码生成库(如 CGLib)等一切会运行时产生新代码的功能都不再可用。
- 对于反射,需要用户在编译期,通过配置文件或编译器参数的形式,明确告知编译器程序代码中哪些方法只通过反射来访问的。
- 用户往往不知道动态生成字节码的具体信息,这些只能由程序去做妥协。默认情况下,每一个 Spring 管理的 Bean 都要用到 CGLib。从 Spring Framework 5.2 开始增加了@proxyBeanMethods 注解来排除对 CGLib 的依赖,仅使用标准的动态代理去增强类。
当然 Spring Native 遇到的问题有很多,且仍然处于试验阶段。以原生方式运行后,启动时间是能够缩短很多,但是程序的运行效率还是若于传统基于 JVM 的方式,且编译成原生程序代码的时间更长。
参考链接
- Spring Native documentation
- GitHub spring-projects-experimental/spring-native
- Get Started with GraalVM
GitHub LeetCode 项目
项目 GitHub LeetCode 全解,欢迎大家 star、fork、merge,共同打造最全 LeetCode 题解!
Java 编程思想-最全思维导图-GitHub 下载链接,需要的小伙伴可以自取~!!!
Spring Native 项目,把 Spring 项目编译成原生程序!的更多相关文章
- 创建Unity新项目并编译成游戏程序
注:本人所使用的Unity版本为:Unity5.3.5f1,所使用的VS版本为:Visual.Studio.2013.Ultimate 折腾了快一个月了,终于有时间做自己的啦,哈哈: ) 步骤一:启动 ...
- idea maven项目要想正常编译成war包,需要做的处理
以及右键项目 - Build(第一次打包成war) (第一次Build) - ReBuild(非第一次打包成war)(非第一次Build) 按照顺序做一到几次,就可以成功编译成war包了(如果rebu ...
- cocos2d-x v3.0的window平台搭建和编译成andriod程序
首先添加这个地址到系统环境变量,path 然后打开CMD,输入如下语句 现在就可以创建一个新项目了 这样一个空的cocos2d-x v3.0的项目就创建好了 接下来编译andriod程序 先在系统环境 ...
- Cocos2d-x 3.x 如何编译成安卓程序
1.安装JDK 2.安装eclipse,安卓官方现在不提供eclipse for android,只好自己配置了.首先安装一个eclipse,在Help——Install New SoftWare中安 ...
- 使用Maven将Hadoop2.2.0源码编译成Eclipse项目
编译环境: OS:RHEL 6.3 x64 Maven:3.2.1 Eclipse:Juno SR2 Linux x64 libprotoc:2.5.0 JDK:1.7.0_51 x64 步骤: 1. ...
- Spring Native实战(畅快体验79毫秒启动springboot应用)
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- spring native 初体验实现 小米控制美的空调
目前关于 spring native 分享的文章还比较少 写这篇文章的主要目前是分享一下自己写的一个 小米控制美的空调 的程序 集成 spring native 过程中碰到的一些问题和解决方法 先放地 ...
- Git下载Spring项目源码并编译为Eclipse
1)当前系统中安装了gradle,如果为安装,可以从:http://www.gradle.org/downloads,,下载完后进行解压到任意盘符,然后增加环境变量GRADLE_HOME,并在环境变量 ...
- Spring+SpringMVC+Mybatis+MAVEN+Eclipse+项目完整环境搭建
1.新建一个Maven项目,创建父项目. 2.创建子项目模块 3.创建javaWeb项目 4.创建后的项目目录结构 5.Maven文件配置 parent父项目pom.xml文件配置 <?xml ...
随机推荐
- Swift in Action
Swift in Action Swift Playgrounds https://apps.apple.com/us/app/swift-playgrounds/id1496833156?mt=12 ...
- PWA & bug
PWA bug https://developer.mozilla.org/zh-CN/docs/Web/Progressive_web_apps https://learning.xgqfrms.x ...
- 聊聊CPU的LOCK指令
本文转载自聊聊CPU的LOCK指令 导语 在多线程操作中,可能最经常被提起的就是数据的可见性.原子性.有序性.不管是硬件方面.软件方面都在这三方面做了很足的工作,才能保证程序的正常运行. 之前发表过一 ...
- c# 全选和批量修改
//全选 function checkAll(){ var items = document.getElementsByTagName("input"); for(var i =0 ...
- WPF 解决内置谷歌浏览器(Cef.ChromiumWebBrowser)在触摸屏无法进行滚动的问题
1.问题描述: 最近在WPF的项目中,需要在控件中嵌套可以浏览特定网页的内容,所以使用了 Cef.ChromiumWebBrowser来解决问题.在执行项目的过程中,主要碰到的问题有: 1.1 当把项 ...
- 微信小程序:数组拼接
一开始用concat进行拼接,总是不行,代码如下: handleItemChange(e){ console.log(e) var itemList = e.detail.value itemList ...
- Linux安装与使用
1.安装 1.1安装VMware 1.1.1VM12版本安装 1)下载:网盘:链接:https://pan.baidu.com/s/1Jnr--KIy3bSTvRhtB8nfiQ 提取码:czna 2 ...
- 一些 html+css 细节
一. input 光标(插入符)颜色 input: { caret-color: #c0c0ff; } 二. 修改 placeholder 颜色 input::placeholder { color: ...
- 绿色城市之地下综合管廊3D可视化平台
前言 现阶段,我国绿色城市建设发展正在热火朝天的进行,面对迅速城镇化建设导致的城市病,需要不断寻求足以丰富城市的资源,以此实现城市绿色化智能化发展,比如改造地下管廊.路灯等城市基础设施. 地下综合管廊 ...
- Mybatis检查SQL注入
Mybatis 的 Mapper.xml 语句中 parameterType 向SQL语句传参有两种方式:#{ } 和 ${ }. 使用#{ }是来防止SQL注入,使用${ }是用来动态拼接参数. 如 ...