结论:
通过buildx可以使用Dockerfile构建多平台镜像;
但是针对java工程的maven Jib插件,可以直接使用其中高版本的实验功能“platforms”构建多平台镜像,不需要修改docker配置。
可使用buildx验证或docker inspect查看镜像的编译平台

交叉编译教程
模拟目标硬件的用户空间
binfmt_misc
程序运行时动态翻译二进制文件

使用交叉编译器

构建多平台 Docker 镜像
Docker 19.03 引入的插件 buildx,构建多平台 Docker 镜像
利用 BuildKit的全部功能扩展了 docker build 的功能

  1. #启用 buildx 插件
  2. #Docker 版本不低于 19.03
  3. #设置环境变量(开启实验特性)
  4. vi /etc/profile
  5. export DOCKER_CLI_EXPERIMENTAL=enabled
  6. source /etc/profile
  7. #验证是否开启
  8. docker buildx version
  9. #github.com/docker/buildx v0.3.1-tp-docker 6db68d029599c6710a32aa7adcba8e5a344795a7
  10. #启用 binfmt_misc
  11. docker run --rm --privileged docker/binfmt:66f9012c56a8316f9244ffd7622d7c21c1f6f28d
  12. #验证是 binfmt_misc 否开启
  13. ls -al /proc/sys/fs/binfmt_misc/
  14. cat /proc/sys/fs/binfmt_misc/qemu-aarch64

从默认的构建器切换到多平台构建器

  1. #创建一个新的构建器
  2. docker buildx create --use --name mybuilder
  3. #启动构建器
  4. docker buildx inspect mybuilder --bootstrap
  5. #查看当前构建器
  6. docker buildx ls
  7. #使用Dockerfile构建本地镜像
  8. docker buildx build -t hello:amd64 --platform=linux/amd64 -o type=docker .
  9. docker buildx build -t hello:arm64 --platform=linux/arm64 -o type=docker .
  10. #运行镜像
  11. #后台运行
  12. docker run --name hello-amd64 -d hello:amd64
  13. docker run --name hello-arm64 -d hello:arm64

Dockerfile:

  1. #amd64
  2. #FROM adoptopenjdk/openjdk8-openj9:alpine-slim
  3. #arm64
  4. FROM adoptopenjdk/openjdk8-openj9:aarch64-ubi-minimal-jre8u-nightly
  5. EXPOSE 8080
  6. VOLUME /tmp
  7. ADD hello.jar /app.jar
  8. ENV JAVA_OPTS="-Dfile.encoding=UTF8 -Duser.timezone=GMT+08"
  9. ENTRYPOINT [ "sh", "-c", "java $JAVA_OPTS -Djava.security.egd=file:/dev/./urandom -jar /app.jar" ]

maven jib插件构建多平台镜像

jib多平台构建参数

jib多平台构建具体操作
若只编译多平台镜像,则此方法无需修改docker相关配置

获取多平台的基础镜像

docker官网搜索可用的多平台镜像,Tags的OS/ARCH会显示具体支持平台
openjdk:8-jdk-alpine

远程镜像查看命令(需要启用Docker CLI实验特性,之前在启用buildx时已经设置相关环境变量):
docker manifest inspect openjdk:8-jdk-alpine

结果:
QQ截图20210321185545.png

修改jib插件配置

实验特性:platforms
from.platforms.platform :配置基础映像的平台以从清单清单中进行选择。

Property Type Default Description
architecture string amd64 The architecture of a base image to select from a manifest list.
os string linux The OS of a base image to select from a manifest list.

pom.xml

  1. <jib-maven-plugin.version>2.8.0</jib-maven-plugin.version>
  2. <plugin>
  3. <groupId>com.google.cloud.tools</groupId>
  4. <artifactId>jib-maven-plugin</artifactId>
  5. <version>${jib-maven-plugin.version}</version>
  6. <configuration>
  7. <from>
  8. <image>${docker.image.from}</image>
  9. <platforms>
  10. <platform>
  11. <architecture>amd64</architecture>
  12. <os>linux</os>
  13. </platform>
  14. <platform>
  15. <architecture>arm64</architecture>
  16. <os>linux</os>
  17. </platform>
  18. </platforms>
  19. </from>
  20. ...
  21. </configuration>
  22. </plugin>

运行maven构建命令

多平台构建只能运行jib:build 推送到镜像仓库

mvn clean package jib:build
构建结果:
QQ截图20210321020229.png
验证命令:docker buildx imagetools inspect 127.0.0.1:5000/demo/snapshots/hello

QQ截图20210321020411.png

可以通过 docker pull 127.0.0.1:5000/demo/snapshots/hello 拉取刚刚创建的镜像了,Docker 将会根据你的 CPU 架构拉取匹配的镜像

启用本地镜像仓库

若没有私有的远程仓库,则可以启用本地仓库

  1. #启用本地镜像仓库
  2. docker pull registry
  3. #启动
  4. docker run -itd -v /root/docker-registry:/data/registry --restart=always --name docker-registry -p 5000:5000 registry:latest
  5. #本地镜像打tag
  6. docker tag a3562aa0b991 127.0.0.1:5000/openjdk:8-jdk-alpine
  7. #推送到本地镜像库
  8. docker push 127.0.0.1:5000/openjdk:8-jdk-alpine

通过命令行指定jib编译平台

pom文件设置

  1. <properties>
  2. <docker.repostory>xxx</docker.repostory>
  3. <docker.registry.name>xxx</docker.registry.name>
  4. <docker.image.from>openjdk:8-jdk-alpine</docker.image.from>
  5. <docker.image.to>xxx</docker.image.to>
  6. <docker.platform.arch>amd64</docker.platform.arch>
  7. </properties>
  8. <build>
  9. <plugin>
  10. <groupId>com.google.cloud.tools</groupId>
  11. <artifactId>jib-maven-plugin</artifactId>
  12. <version>${jib-maven-plugin.version}</version>
  13. <configuration>
  14. <from>
  15. <image>${docker.image.from}</image>
  16. <platforms>
  17. <platform>
  18. <architecture>${docker.platform.arch}</architecture>
  19. <os>linux</os>
  20. </platform>
  21. </platforms>
  22. </from>
  23. <to>
  24. <image>
  25. ${docker.repostory}/${docker.registry.name}/${docker.image.to}/${project.artifactId}
  26. </image>
  27. </to>
  28. ...
  29. </configuration>
  30. </plugin>
  31. </build>

arm64平台编译

mvn clean package jib:build -Ddocker.platform.arch=arm64 -Djib.to.tags=arm64

拉取生成的镜像后,使用docker inspect 查看结果:
docker inspect 127.0.0.1:5000/demo/snapshots/hello:arm64
QQ截图20210321184400.png


x86-64平台编译

mvn clean package jib:build -Ddocker.platform.arch=amd64 -Djib.to.tags=amd64

拉取生成的镜像后,使用docker inspect 查看结果:
docker inspect 127.0.0.1:5000/demo/snapshots/hello:amd64
QQ截图20210321184522.png

参考文档:

buildx教程
buildx官方文档
jib操作教程1

jib操作教程2