Java程序制做Docker Image推荐方案

原文地址html

本文源代码是一个spring-boot应用(在 https://github.com/chanjarste... ),不过本例子适用于全部Java应用。java

要求

这里先给出一些Docker Image制做的要求,以后咱们再看怎么作。mysql

  1. 制做过程要融合在项目构建过程当中
  2. 使用官方Image做为基础Image
  3. 设定正确的时区
  4. Container内的程序以非root用户启动
  5. 指定Web程序的端口
  6. 可以传递JVM参数、Java System Properties、程序自定义的参数

下面具体讲一下具体怎么作到以上几点:git

制做过程要融合在项目构建过程当中

这里推荐使用Spotify的dockerfile-maven-plugin,理由是这个plugin用起来最简单且容易掌握。github

该plugin的本质上是你写一个Dockerfile(关于Dockerfile的具体写法请参照官方文档),这个plugin把一些参数传递进去来帮助你构建Docker Image。spring

所以只要你会写Dockerfile,就会使用这个plugin,它没有加入任何额外的概念。sql

使用官方Image做为基础Image

Java的基础镜像应该在openjdk repository里寻找,而不是在已通过时的java repository里找。docker

openjdk repository提供了各类各样的image tags看起来眼花缭乱,可是本质上来讲就这么几个:shell

  • openjdk:<version>
  • openjdk:<version>-slim
  • openjdk:<version>-alpine

关于<version>通常来讲指定大版本号就好了,好比你能够在Dockerfile这样写:数据库

FROM openjdk:8-alpine

从尺寸上来说,alpine最小、slim稍大、默认的最大。因此应该尽量的使用alpine版本的,若是发现程序的运行环境缺乏某些东西,那么尝试用slim版本或者默认版本。就目前的经验来说:

  • 若是须要操做系统字体库,那么就得使用slim版本或者默认版本。须要操做系统字体库的程序例如:图片验证码、PDF导出。
  • 若是须要某些Linux标准的动态/静态链接库,那么在alpine版本不行的状况下,尝试slim版本或默认版本。由于alpine版本是一个及其精简的Linux,它删除了不少东西。

设定正确的时区

几乎全部的Docker Image的时区都是UTC,咱们须要给咱们本身制做的Docker Image设定时区:

ENV TZ=Asia/Shanghai
RUN set -eux; \
    ln -snf /usr/share/zoneinfo/$TZ /etc/localtime; \
    echo $TZ > /etc/timezone

若是是alpine系列的则要安装tzdata:

ENV TZ=Asia/Shanghai
RUN set -eux; \
    apk add --no-cache --update tzdata; \
    ln -snf /usr/share/zoneinfo/$TZ /etc/localtime; \
    echo $TZ > /etc/timezone

关于数据库时区的相关内容能够见:

Container内的程序以非root用户启动

在Docker Image内部,咱们应该使用非root用户启动程序,这须要新建用户。

若是你用的是openjdk:<version>-alpine新建用户命令是这样的:

RUN set -eux; \
    addgroup --gid 1000 java-app; \
    adduser -S -u 1000 -g java-app -h /home/java-app/ -s /bin/sh -D java-app;

若是你用的是openjdk:<version>-slim或者openjdk:<version>新建用户命令是这样的:

RUN set -eux; \
    addgroup --gid 1000 java-app; \
    adduser --system --uid 1000 --gid 1000 --home=/home/java-app/ --shell=/bin/sh --disabled-password java-app;

而后使用Dockerfile USER指令

USER java-app

# 后面的指令就都是以java-app用户身份执行了

指定Web程序的接口

对于联网应用而言,必须在Dockerfile中指定暴露的端口,不然该端口没法映射。

EXPOSE 8080

可以传递JVM参数、Java System Properties、程序自定义的参数

咱们须要可以在启动Docker Image的时候将一些参数传递进去:

  • JVM参数
  • Java System Properties
  • 程序启动参数

这里就须要参考Dockerfile best practiceDocker ENTRYPOINT了。

样例项目拆解

目录结构

全部与程序相关的东西都存放在/home/java-app/下:

/home/java-app
   ├── docker-entrypoint.sh
   ├── lib
   │   └── java-app.jar
   ├── etc
   ├── logs
   └── tmp
  • docker-entrypoint.sh,启动脚本
  • lib,存放JAR包
  • lib/java-app.jar,程序JAR包
  • etc,存放配置文件
  • logs,存放日志文件
  • tmp,存放临时文件

构建Image的方法

mvn clean package dockerfile:build

运行

普通启动,而后访问http://localhost:8080

docker run -p 8080:8080 chanjarster/dockerfile-java-examples-1:1.0-SNAPSHOT

设定JVM参数/System Properties,使用JAVA_OPTS环境变量:

docker run -p 8080:8080 -e JAVA_OPTS='-Xmx128M -Xms128M -Dabc=xyz -Ddef=uvw' chanjarster/dockerfile-java-examples-1:1.0-SNAPSHOT

提供程序运行参数,在后面直接添加便可:

docker run -p 8080:8080 chanjarster/dockerfile-java-examples-1:1.0-SNAPSHOT --debug

2018-12-27更新

docker-entrypoint.sh里启动Java进程时,使用exec /usr/bin/java ...这种形式,保证进程PID=1,这样在进程可以在docker stop时收到SIGTERM
详见:docker stop

参考文档

相关文章
相关标签/搜索