Spring Boot: 3.2.0,Kotlin: 1.9.21,构建映像: maven:3.9.5-amazoncorretto-21,运行映像: amazoncorretto:17
我有一个非常简单的 Spring Boot 项目
@SpringBootApplication
class DemoApplication
fun main(args: Array<String>) {
runApplication<DemoApplication>(*args)
listOf(1,2,3,4).removeLast()
}
pom 看起来像
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>3.2.0</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.example</groupId>
<artifactId>demo</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>demo</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>17</java.version>
<kotlin.version>1.9.21</kotlin.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.jetbrains.kotlin</groupId>
<artifactId>kotlin-reflect</artifactId>
</dependency>
<dependency>
<groupId>org.jetbrains.kotlin</groupId>
<artifactId>kotlin-stdlib</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<sourceDirectory>${project.basedir}/src/main/kotlin</sourceDirectory>
<testSourceDirectory>${project.basedir}/src/test/kotlin</testSourceDirectory>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<groupId>org.jetbrains.kotlin</groupId>
<artifactId>kotlin-maven-plugin</artifactId>
<configuration>
<args>
<arg>-Xjsr305=strict</arg>
</args>
<compilerPlugins>
<plugin>spring</plugin>
</compilerPlugins>
<jvmTarget>${java.version}</jvmTarget>
</configuration>
<dependencies>
<dependency>
<groupId>org.jetbrains.kotlin</groupId>
<artifactId>kotlin-maven-allopen</artifactId>
<version>${kotlin.version}</version>
</dependency>
</dependencies>
</plugin>
</plugins>
</build>
</project>
注意 Kotlin 插件配置中的
<jvmTarget>${java.version}</jvmTarget>
指向 Java 17。
FROM maven:3.9.5-amazoncorretto-21 AS builder
RUN mkdir "/app"
WORKDIR /app
COPY src ./src
COPY pom.xml .
RUN mvn clean package
FROM amazoncorretto:17
COPY --from=builder /app/target/demo-0.0.1-SNAPSHOT.jar ./app.jar
ENTRYPOINT ["java", "-jar", "app.jar"]
现在,如果我构建并运行它
docker build -t demo .
docker run demo
我收到以下异常:
INFO 1 --- [ main] com.example.demo.DemoApplicationKt : Starting DemoApplicationKt v0.0.1-SNAPSHOT using Java 17.0.9 with PID 1 (/app.jar started by root in /)
Exception in thread "main" java.lang.reflect.InvocationTargetException
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:77)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:568)
at org.springframework.boot.loader.launch.Launcher.launch(Launcher.java:91)
at org.springframework.boot.loader.launch.Launcher.launch(Launcher.java:53)
at org.springframework.boot.loader.launch.JarLauncher.main(JarLauncher.java:58)
Caused by: java.lang.NoSuchMethodError: 'java.lang.Object java.util.List.removeLast()'
at com.example.demo.DemoApplicationKt.main(DemoApplication.kt:12)
如果您检查与
removeLast()
相关的文档,它是可用版本 21
删除并返回该集合的最后一个元素(可选操作)。 投掷: NoSuchElementException – 如果此集合为空 UnsupportedOperationException – 如果此集合实现不支持此操作 实施要求: 如果此List不为空,则此接口中的实现将返回调用remove(size() - 1)的结果。否则,它会抛出 NoSuchElementException。 自:21
显然,Kotlin 插件忽略了
<jvmTarget>
配置。如果我将构建映像更改为 maven:3.9.5-amazoncorretto-17
,问题就会消失。
这是 Kotlin 插件中的错误吗?
这不是一个错误。它在KT-64593 Kotlin maven插件忽略jvmTarget中进行了解释。
我错过了添加编译器参数 -Xjdk-release=17。
Kotlin 中的-Xjdk-release 类似于 --release JEP 247。
就我而言,我只设置了
jvmTarget
,这相当于
javac --target 17
Java 中。
请注意,在上面我没有指定--source。默认行为是(如果缺少)默认为当前 Java SE 版本(您正在使用的版本)。