当前位置:   article > 正文

使用 Spring Native 毫秒级启动 SpringBoot 项目,并且大量减少占用的内存_maven package spring native

maven package spring native

Spring Native是什么

近几年“原生”一词一直泛滥在云计算、边缘计算等领域中,而原生宠幸的语言也一直都是Golang,Rust等脱离Sandbox运行的开发语言。Java得益于上世纪流行的一次编译,到处执行的理念,流行至今,但也因为这个原因,导致Java程序脱离不了JVM运行环境,使得不那么受原生程序的青睐。在云原生泛滥的今天,臃肿的JVM使Java应用程序对比其他语言显得无比的庞大,各路大神也想了很多方式让Java变的更“原生”。

Spring Native 为使用GraalVM 原生映像编译器将 Spring 应用程序编译为原生可执行文件提供 beta 支持,以提供通常设计为打包在轻量级容器中的原生部署选项。实际上,目标是在这个新平台上支持几乎未修改的 Spring Boot 应用程序。

配置环境

  1. OS:Windows10 21H2
  2. IDE:IntelliJ IDEA 2022.1.3
  3. JDK:graalvm-ce-java11-22.2.0
  4. Maven:3.5.4
  5. Docker Desktop for Windows: 4.12.0
  6. Spring Boot:2.7.4
  7. Spring Native0.12.1

使用 Spring Native 的应用程序应该使用 Java 11 或 Java 17 编译。

构建 Spring Boot 原生应用程序有两种主要方法:

  • 使用Spring Boot Buildpacks 支持生成包含本机可执行文件的轻量级容器。

  • 使用GraalVM 原生镜像 Maven 插件支持生成原生可执行文件。

方法1就是在SpringBoot2.3后,可以使用spring-boot-maven-plugin插件来构建docker镜像,使用mvn spring-boot:build-image命令结合Docker的API来实现Spring Boot 原生应用程序的构建,成功执行后会直接生成一个docker镜像,然后run这个镜像就可以了,不用我们再写Dockerfile了,相关的参数配置都在pom.xml中配置(该插件的configuration标签下,和fabric8或spotify的docker-maven-plugin很相似)。

方法2不需要安装docker,但要安装Visual Studio,然后执行mvn -Pnative package命令后会生成一个可执行文件(.exe),运行即可。

主要区别如下

1 环境依赖不同

  • 方法1需要安装Docker

  • 方法2需要安装Visual Studio(需要用到部分单个组件:2个MSVC,1个Windows 10 SDK)

2 执行的maven命令不同

  • 方法1是mvn spring-boot:build-image

  • 方法2是mvn -Pnative package

因为每个微服务使用Docker部署而不是exe文件,所以方法1正好符合我的需求,所以后文使用Spring Boot Buildpacks的方式构建Spring Boot原生应用程序。

安装Graal VM

官方下载地址:

https://www.graalvm.org/downloads/

配置环境变量

检验是否安装成功

安装native-image

打开新的cmd,输入以下命令,等待安装

gu install native-image

安装 Docker Desktop for Windows

具体步骤略,按照官方文档操作即可:https://docs.docker.com/desktop/windows/install/

创建Springboot项目

完整的pom如下

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3.  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  4.  <modelVersion>4.0.0</modelVersion>
  5.  <parent>
  6.   <groupId>org.springframework.boot</groupId>
  7.   <artifactId>spring-boot-starter-parent</artifactId>
  8.   <version>2.7.4</version>
  9.   <relativePath/> <!-- lookup parent from repository -->
  10.  </parent>
  11.  <groupId>com.example</groupId>
  12.  <artifactId>spring-native</artifactId>
  13.  <version>0.0.1-SNAPSHOT</version>
  14.  <name>spring-native</name>
  15.  <description>spring-native</description>
  16.  <properties>
  17.   <java.version>11</java.version>
  18.   <repackage.classifier/>
  19.   <spring-native.version>0.12.1</spring-native.version>
  20.  </properties>
  21.  <dependencies>
  22.   <dependency>
  23.    <groupId>org.springframework.boot</groupId>
  24.    <artifactId>spring-boot-starter-web</artifactId>
  25.   </dependency>
  26.   <dependency>
  27.    <groupId>org.springframework.experimental</groupId>
  28.    <artifactId>spring-native</artifactId>
  29.    <version>${spring-native.version}</version>
  30.   </dependency>
  31.   <dependency>
  32.    <groupId>org.projectlombok</groupId>
  33.    <artifactId>lombok</artifactId>
  34.    <optional>true</optional>
  35.   </dependency>
  36.   <dependency>
  37.    <groupId>org.springframework.boot</groupId>
  38.    <artifactId>spring-boot-starter-test</artifactId>
  39.    <scope>test</scope>
  40.   </dependency>
  41.  </dependencies>
  42.  <build>
  43.   <plugins>
  44.    <plugin>
  45.     <groupId>org.springframework.boot</groupId>
  46.     <artifactId>spring-boot-maven-plugin</artifactId>
  47.     <configuration>
  48.      <excludes>
  49.       <exclude>
  50.        <groupId>org.projectlombok</groupId>
  51.        <artifactId>lombok</artifactId>
  52.       </exclude>
  53.      </excludes>
  54.      <classifier>${repackage.classifier}</classifier>
  55.      <image>
  56.       <builder>paketobuildpacks/builder:tiny</builder>
  57.       <env>
  58.        <BP_NATIVE_IMAGE>true</BP_NATIVE_IMAGE>
  59.        <BPE_DELIM_JAVA_TOOL_OPTIONS xml:space="preserve"> </BPE_DELIM_JAVA_TOOL_OPTIONS>
  60.        <BPE_APPEND_JAVA_TOOL_OPTIONS>-Xms128m</BPE_APPEND_JAVA_TOOL_OPTIONS>
  61.        <BPE_APPEND_JAVA_TOOL_OPTIONS>-Xmx128m</BPE_APPEND_JAVA_TOOL_OPTIONS>
  62.        <BPE_APPEND_JAVA_TOOL_OPTIONS>-Xss256k</BPE_APPEND_JAVA_TOOL_OPTIONS>
  63.        <BPE_APPEND_JAVA_TOOL_OPTIONS>-XX:ParallelGCThreads=2</BPE_APPEND_JAVA_TOOL_OPTIONS>
  64.        <BPE_APPEND_JAVA_TOOL_OPTIONS>-XX:+PrintGCDetails</BPE_APPEND_JAVA_TOOL_OPTIONS>
  65.       </env>
  66.      </image>
  67.     </configuration>
  68.    </plugin>
  69.    <plugin>
  70.     <groupId>org.springframework.experimental</groupId>
  71.     <artifactId>spring-aot-maven-plugin</artifactId>
  72.     <version>${spring-native.version}</version>
  73.     <executions>
  74.      <execution>
  75.       <id>test-generate</id>
  76.       <goals>
  77.        <goal>test-generate</goal>
  78.       </goals>
  79.      </execution>
  80.      <execution>
  81.       <id>generate</id>
  82.       <goals>
  83.        <goal>generate</goal>
  84.       </goals>
  85.      </execution>
  86.     </executions>
  87.    </plugin>
  88.   </plugins>
  89.  </build>
  90.  <repositories>
  91.   <repository>
  92.    <id>spring-releases</id>
  93.    <name>Spring Releases</name>
  94.    <url>https://repo.spring.io/release</url>
  95.    <snapshots>
  96.     <enabled>false</enabled>
  97.    </snapshots>
  98.   </repository>
  99.  </repositories>
  100.  <pluginRepositories>
  101.   <pluginRepository>
  102.    <id>spring-releases</id>
  103.    <name>Spring Releases</name>
  104.    <url>https://repo.spring.io/release</url>
  105.    <snapshots>
  106.     <enabled>false</enabled>
  107.    </snapshots>
  108.   </pluginRepository>
  109.  </pluginRepositories>
  110. </project>

本文介绍的是Spring Native0.12.1版本,其对应的Spring Boot版本必须是2.7.4,以上只是一个最基本的配置案例,实际开发中还需要在spring-boot-maven-plugin插件的configuration标签下配置其他许许多多的参数。

例如docker远程的地址和证书的路径、jvm调优参数、配置文件指定、docker镜像名端口仓库地址等等,最好的方法就是看spring-boot-maven-plugin的官方文档,这里以配置jvm参数为例

配置官方文档:

https://docs.spring.io/spring-boot/docs/2.6.2/maven-plugin/reference/htmlsingle/#build-image

为了测试我这里配置Springboot项目端口为8888,并且添加了一个获取当前日期的API。

执行maven命令

  1. mvn clean
  2. mvn '-Dmaven.test.skip=true' spring-boot:build-image

下载完相关依赖后,电脑就会开始构建编译项目,查看任务管理器可以发现在构建和编译过程中CPU利用率很高,内存使用量也很高,根据电脑配置和网络状态不同使用的构建时间也不相同。

构建成功如下图:

创建并运行容器

查看所有镜像docker images

spring-native就是构建的镜像

创建并运行容器

docker run -id -p 8888:8888 --name=native-app spring-native:0.0.1-SNAPSHOT 59178df9f49b

Docker Desktop查看日志,发现应用成功启动,启动仅耗时31ms,果然印证了Spring Native启动是毫秒级别这句话。

Docker Desktop查看占用内存,仅46M左右。

成功调用接口

不使用Spring Native启动应用,使用java -jar来启动项目

启动耗时1.5秒,占用内存高达238M。

最后附上Spring Native官方文件:

https://docs.spring.io/spring-native/docs/current/reference/htmlsingle/index.html

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/383287
推荐阅读
相关标签
  

闽ICP备14008679号