Docker部署Golang项目及镜像优化实战
今天在部署公司的项目时, 了解到目标服务器有Docker环境,于是就想着基于Docker来部署项目
这篇文章记录了一次完整的构建镜像到优化的完整过程.
基础环境
Docker
编辑Dockerfile文件
基于简单,快速构建部署的原则,我选择在构建docker镜像时编译程序.即 上传项目源代码,通过Dockerfile命令声明完成编译,再运行二进制文件.先展示我的文件目录
main文件,跟需要的comfigs文件夹 都在项目跟目录. (我的程序启动时需要 load configs/目录下的配置文件)
那么,我们的Dockerfile文件就直接放在项目 根目录了
Dockerfile文件内容
1 #基础镜像,这也是我本地的golang版本 2 FROM golang:1.15.10 3 4 #环境变量 5 ENV GO111MODULE=on 6 ENV GOPROXY=https://goproxy.cn,direct 7 8 #指定操作目录 9 WORKDIR /go/src/labs.api 10 #复制根目录内所有源码文件到操作目录下 11 COPY . . 12 13 #编译 14 RUN GOOS=linux CGO_ENABLED=1 GOARCH=amd64 go build -installsuffix cgo . 15 16 #暴露的端口 17 EXPOSE 18888 18 19 #运行 labs.api是我的项目名称 20 CMD ["./labs.api"]
编译并运行
编译
1 #编译 2 docker build -t labs.api:v1.0 . 3 4 #运行 5 docker run -p 18888:18888 --name labs.api labs.api:v1.0
程序正常运行,那么我们文章标题里的 优化从何说起呢?
接下来我们看下镜像文件
1.34GB记住这个数值
这是我们刚刚构建的镜像文件,一个可执行二进制只需要几十M,为什么镜像文件会这么大.
是因为我们构建项目所需要的基础镜像过于大.
那么我们只需要把可执行二进制文件放到一个小的基础运行环境的镜像中即可
优化
优化Dockerfile
1 #基础镜像,这也是我本地的golang版本 2 FROM golang:1.15.10 as builder 3 4 #环境变量 5 ENV GO111MODULE=on 6 ENV GOPROXY=https://goproxy.cn,direct 7 8 #指定工作目录 9 WORKDIR /go/src/labs.api 10 #复制根目录内所有源码文件到工作目录下 11 COPY . . 12 13 #编译 14 RUN GOOS=linux CGO_ENABLED=1 GOARCH=amd64 go build -installsuffix cgo . 15 16 #基础镜像,运行环境 17 FROM ubuntu:18.04 as runner 18 19 #指定工作目录 20 WORKDIR /go/app 21 #复制二进制执行文件和必要的Configs目录到工作目录 22 COPY --from=builder /go/src/labs.api/labs.api . 23 COPY --from=builder /go/src/labs.api/configs ./configs 24 25 #暴露的端口 26 EXPOSE 18888 27 28 #运行 labs.api是我的项目名称 29 CMD ["./labs.api"]
再次编译运行,正常启动!
然后查看镜像文件大小
相差1.1个G
总结
我们在使用Docker部署项目时,只需要把 编译及运行环境分开,就能够减小镜像的大小.并且效果非常令人满意.
具体原理可以看一下Docker的多级构建介绍,关于多级构建镜像的原理.
简单来说就是 单级构建只有一个“From”语句,而在多级构建中,有多个“From”,每个“From”构成一级。就像我的项目里的Dockerfile文件有两个“From”,是一个二级构建。每一级都可以根据需要选择适合自己的基础(base)镜像来构造本级镜像。每级镜像完成之后,下一级镜像可选择只保留上一级构建中对自己有用的最终文件(构建的二进制文件),而删除所有的中间产物,这样就大大节省了空间。
到此结束。
有问题,或者建议请留言,谢谢。