在《Go语言实战指南》的“构建与部署”章节中,Docker容器化部署是现代 Go 应用发布与运维的重要方式之一。通过将 Go 应用打包进 Docker 容器,我们可以实现“构建一次,处处运行”,无论目标服务器操作系统如何,都可以保持一致的运行环境。
Go 原生编译出的可执行文件已具备高度可移植性,但在实际部署中仍会面临以下挑战:
而 Docker 作为容器化技术的代表,能够封装应用及其依赖,提升:
// main.go
package main
import (
"fmt"
"net/http"
)
func hello(w http.ResponseWriter, r *http.Request) {
fmt.Fprintln(w, "Hello, Dockerized Go!")
}
func main() {
http.HandleFunc("/", hello)
http.ListenAndServe(":8080", nil)
}
有两种方式:
# 第一阶段:构建阶段
FROM golang:1.21 AS builder
WORKDIR /app
COPY . .
RUN go mod tidy
RUN CGO_ENABLED=0 GOOS=linux go build -o app
# 第二阶段:运行阶段
FROM alpine:latest
RUN apk --no-cache add ca-certificates
WORKDIR /root/
COPY --from=builder /app/app .
EXPOSE 8080
CMD ["./app"]
说明:
CGO_ENABLED=0
禁用 cgo,确保在无 libc 的容器中运行正常docker build -t my-go-app .
docker run -d -p 8080:8080 my-go-app
访问 http://localhost:8080,即可看到输出。
.dockerignore
排除无关文件创建 .dockerignore
文件:
.git
node_modules
*.log
Dockerfile
README.md
避免无用文件进入镜像,缩小体积、提高构建速度。
在 Dockerfile 中使用环境变量:
ENV APP_ENV=production
运行时也可指定:
docker run -e APP_ENV=development -p 8080:8080 my-go-app
Go 应用中的日志应直接输出到标准输出(stdout),方便 Docker 守护进程或云原生平台(如 Kubernetes)统一收集日志。
log.Println("Service started...")
在 Dockerfile
中加入 HEALTHCHECK
指令:
HEALTHCHECK --interval=30s --timeout=10s \
CMD wget --spider http://localhost:8080 || exit 1
如需连接数据库、Redis 等服务,推荐使用 docker-compose.yml
管理多个容器:
version: "3"
services:
app:
build: .
ports:
- "8080:8080"
redis:
image: redis:alpine
然后一键启动:
docker-compose up --build
go mod download
单独放在 COPY 前。golang:1.21.5
代替 golang:latest
。优势 | 说明 |
可移植性强 | 一次构建,处处运行 |
简化部署流程 | 不依赖操作系统环境 |
易于集成自动化 | 与 CI/CD 系统兼容 |
快速回滚 | 镜像级版本控制 |
Docker + Go 是部署现代云原生应用的黄金组合。