Anda telah mencipta aplikasi Spring Boot. Ia berfungsi hebat pada mesin tempatan anda dan kini, anda perlu menggunakan aplikasi itu di tempat lain. Pada sesetengah platform, anda boleh menyerahkan terus fail balang dan ia akan digunakan. Di sesetengah tempat, anda boleh memutar mesin maya, memuat turun kod sumber di sana, membinanya dan menjalankannya. Tetapi, kebanyakan masa anda perlu menggunakan aplikasi menggunakan bekas. Selalunya, Docker digunakan untuk membina dan menjalankan imej dalam bekas. Selain itu, apabila anda memuat naik fail balang ke beberapa platform, aplikasi dijalankan di dalam bekas di bawah hud.
Jadi, dalam blog ini, kita akan melihat 3 cara berbeza untuk membina imej Docker untuk aplikasi Spring Boot yang diberikan. Jom mulakan:
Cara yang naif dan tidak mencukupi untuk membina imej Docker untuk sebarang aplikasi adalah dengan menggunakan Dockerfile mudah yang menyalin fail jar di dalam imej dan menjalankannya menggunakan arahan java -jar.
Berikut ialah Dockerfile yang boleh anda letakkan pada akar projek:
FROM eclipse-temurin:21-jre-ubi9-minimal ARG JAR_FILE COPY ${JAR_FILE} application.jar ENTRYPOINT ["java", "-jar", "/application.jar"]
Kami telah menetapkan satu hujah JAR_FILE yang merupakan lokasi fail balang untuk digunakan.
Selepas mencipta Dockerfile di atas, langkah di bawah digunakan untuk mencipta imej Docker:
Bina fail balang untuk projek Spring Boot:
./gradlew bootJar # For Gradle build system
ATAU
./mvnw spring-boot:build-jar # For Maven build system
Gunakan Fail Docker untuk membina imej Docker menggunakan fail balang terkini. Dalam arahan di bawah gantikan {IMAGE_NAME} dengan nama imej yang diperlukan dan {JAR_FILE} dengan laluan ke fail jar yang dijana. Nama imej mengandungi teg juga, seperti - mycompany/product-service:0.0.1-SNAPSHOT:
docker build --build-arg JAR_FILE={JAR_FILE} --tag {IMAGE_NAME} .
Sahkan jika imej Docker dibina menggunakan arahan berikut. Anda sepatutnya dapat melihat imej dengan nama yang dinyatakan dalam arahan di atas:
docker images
Walaupun boleh dan mudah untuk membungkus balang uber Spring Boot sebagai imej Docker (seperti yang dinyatakan dalam kaedah sebelumnya), terdapat banyak kelemahan untuk menyalin dan menjalankan balang gemuk sebagaimana adanya dalam imej Docker. Contohnya,
Memandangkan kami menyusun kod kami lebih kerap daripada menaik taraf versi Spring Boot, adalah lebih baik untuk memisahkan perkara lebih sedikit. Jika kita meletakkan fail jar tersebut (yang jarang ditukar) dalam lapisan sebelum lapisan aplikasi, maka Docker selalunya perlu menukar hanya lapisan bawah dan boleh memilih yang lain daripada cachenya.
Untuk mencipta imej Docker berlapis, kita perlu mencipta balang berlapis terlebih dahulu. Pada masa kini, ia didayakan secara lalai dalam Gradle dan Maven. Anda boleh mendayakan atau melumpuhkan gelagat balang berlapis menggunakan tetapan berikut:
// build.gradle tasks.named("bootJar") { layered { enabled = false } }
// build.gradle.kts tasks.named<BootJar>("bootJar") { layered { enabled.set(false) } }
<!-- pom.xml --> <project> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <layers> <enabled>true</enabled> </layers> </configuration> </plugin> </plugins> </build> </project>
Anda juga boleh menala cara lapisan dibuat. Lihat dokumentasi untuk konfigurasi gradle atau maven.
Di bawah ialah Dockerfile, yang boleh digunakan untuk memanfaatkan balang berlapis dan untuk mencipta imej Docker berlapis bagi aplikasi Spring Boot.
# Perform the extraction in a separate builder container FROM eclipse-temurin:21-jre-ubi9-minimal AS builder WORKDIR /builder # This points to the built jar file in the target folder # Adjust this to 'build/libs/*.jar' if you're using Gradle ARG JAR_FILE=target/*.jar # Copy the jar file to the working directory and rename it to application.jar COPY ${JAR_FILE} application.jar # Extract the jar file using an efficient layout RUN java -Djarmode=tools -jar application.jar extract --layers --destination extracted # This is the runtime container FROM eclipse-temurin:21-jre-ubi9-minimal WORKDIR /application # Copy the extracted jar contents from the builder container into the working directory in the runtime container # Every copy step creates a new docker layer # This allows docker to only pull the changes it really needs COPY --from=builder /builder/extracted/dependencies/ ./ COPY --from=builder /builder/extracted/spring-boot-loader/ ./ COPY --from=builder /builder/extracted/snapshot-dependencies/ ./ COPY --from=builder /builder/extracted/application/ ./ # Start the application jar - this is not the uber jar used by the builder # This jar only contains application code and references to the extracted jar files # This layout is efficient to start up and CDS friendly ENTRYPOINT ["java", "-jar", "application.jar"]
Langkah-langkah untuk membina imej Docker berlapis adalah sama seperti membina imej Docker asas. Sila rujuk di sana.
Bagaimana jika saya memberitahu anda bahawa anda boleh mencipta imej Docker tanpa membuat fail Docker? Kita boleh membina imej docker terus daripada pemalam Gralde atau Maven menggunakan Cloud Native Buildpacks. Sesetengah platform (seperti Heroku atau Cloud Foundry) menggunakan Buildpacks untuk menukar fail balang yang disediakan kepada imej boleh jalan.
Spring Boot termasuk sokongan buildpack secara langsung untuk Maven dan Gradle. Kami tidak perlu memasukkan sebarang pemalam tambahan. Jalankan sahaja arahan di bawah:
./gradlew bootBuildImage # For gradle build system
ATAU
./mvnw spring-boot:build-image # For maven build system
Arahan di atas menjana imej dengan nama lalai {PROJECT_NAME}:${PROJECT_VERSION}. Jika anda ingin mengkonfigurasi nama imej yang dijana, anda boleh mengikuti langkah di bawah:
Kami boleh mengkonfigurasi tugas bootBuildImage untuk menetapkan nama imej, seperti ini:
// For build.gradle.kts val imagePrefix = "javarush" val dockerImageName = "docker-example" tasks.named<BootBuildImage>("bootBuildImage") { imageName.set("${imagePrefix}/${dockerImageName}:${version}") }
// For build.gradle def imagePrefix = "javarush" def dockerImageName = "docker-example" tasks.named("bootBuildImage") { imageName = "${imagePrefix}/${dockerImageName}:${version}" }
Kami boleh mengkonfigurasi spring-boot-maven-plugin untuk menggunakan nama imej lain, seperti ini:
<properties> <imagePrefix>javarush</imagePrefix> </properties> ... <project> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <image> <name>${imagePrefix}/${project.artifactId}:${project.version}</name> </image> </configuration> </plugin> </plugins> </build> </project>
Kita juga boleh menentukan nama imej semasa menjalankan arahan untuk membina imej.
./gradlew bootBuildImage --imageName=javarush/docker-example:1.0.0 # For grade build system ./mvnw spring-boot:build-image -Dspring-boot.build-image.imageName=javarush/docker-example:1.0.0 # For maven build system
You can see the documentation to further configure Gradle or Maven plugin.
This is my go-to method to create a Docker image for any Spring Boot application.
Once you create a docker image, you need to make sure that it works as expected. After you make sure that the image is created, you can directly run it using the docker run command. For example,
docker run -p "8080:8080" {IMAGE_NAME}
But, this is not how images are used in production applications. Docker Compose is used to run and manage multiple docker images.
In this blog, we have seen how to build Docker images for Spring Boot applications using different methods. Being able to build docker images for your apps is a must skill to know because the image is what gets delivered. Thanks for reading the article till the end. I appreciate it. I will meet you in the next one. As always, all feedback and suggestions are welcome.
Atas ialah kandungan terperinci Mencipta Imej Docker bagi Aplikasi Spring Boot menggunakan Buildpacks. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!