docker

关注公众号 jb51net

关闭
首页 > 网站技巧 > 服务器 > 云和虚拟化 > docker > Docker容器化

从入门到精通详解Docker容器化的实战指南

作者:lbb 小魔仙

在微服务时代,Docker已成为开发者的必备技能,本文将带你从零开始,通过实战案例掌握Docker的核心技术,包括容器管理、镜像构建、Docker Compose编排等,附完整可运行代码

前言

你是否遇到过这样的困扰:

Docker的出现完美解决了这些问题!它通过容器化技术,将应用程序及其依赖打包成一个轻量级、可移植的容器,实现"一次构建,到处运行"。

本文将从Docker的基础概念讲起,通过多个实战案例带你掌握容器化技术,最终能够独立完成复杂应用的Docker化部署。

一、Docker核心概念

1.1 什么是Docker

Docker是一个开源的容器化平台,它允许开发者将应用及其依赖打包到一个轻量级、可移植的容器中。

核心优势:

1.2 核心组件对比

组件说明类比
镜像(Image)只读的模板,包含运行应用所需的一切类似于面向对象编程中的"类"
容器(Container)镜像的运行实例类似于"对象"
仓库(Registry)存储和分发镜像的地方类似于代码仓库(如GitHub)
Dockerfile构建镜像的脚本类似于Makefile或CMakeLists.txt

1.3 Docker vs 虚拟机

┌─────────────────────────────────────────────────┐
│                 虚拟机架构                        │
├─────────────────────────────────────────────────┤
│  应用程序A    应用程序B    应用程序C               │
├─────────────────────────────────────────────────┤
│  Guest OS    Guest OS    Guest OS               │
├─────────────────────────────────────────────────┤
│              Hypervisor                         │
├─────────────────────────────────────────────────┤
│                 Host OS                         │
├─────────────────────────────────────────────────┤
│                 硬件                             │
└─────────────────────────────────────────────────┘

┌─────────────────────────────────────────────────┐
│                 Docker架构                       │
├─────────────────────────────────────────────────┤
│  应用程序A    应用程序B    应用程序C               │
├─────────────────────────────────────────────────┤
│              Docker Engine                      │
├─────────────────────────────────────────────────┤
│                 Host OS                         │
├─────────────────────────────────────────────────┤
│                 硬件                             │
└─────────────────────────────────────────────────┘

性能对比:

二、Docker安装与配置

2.1 Windows安装

# 1. 下载Docker Desktop
# 访问 https://www.docker.com/products/docker-desktop/

# 2. 安装后验证
docker --version
docker-compose --version

# 3. 启动Docker Desktop

2.2 Linux安装

# Ubuntu/Debian
curl -fsSL https://get.docker.com -o get-docker.sh
sudo sh get-docker.sh

# 将当前用户添加到docker组
sudo usermod -aG docker $USER

# 重新登录或执行
newgrp docker

# 验证安装
docker --version
docker run hello-world

2.3 配置镜像加速器

# 编辑Docker配置文件
sudo mkdir -p /etc/docker
sudo tee /etc/docker/daemon.json <<-'EOF'
{
  "registry-mirrors": [
    "https://docker.mirrors.ustc.edu.cn",
    "https://hub-mirror.c.163.com"
  ]
}
EOF

# 重启Docker服务
sudo systemctl daemon-reload
sudo systemctl restart docker

三、Docker基础命令

3.1 镜像操作

# 搜索镜像
docker search nginx

# 拉取镜像
docker pull nginx:latest
docker pull python:3.10-slim

# 查看本地镜像
docker images
docker image ls

# 删除镜像
docker rmi nginx:latest
docker image prune -a  # 删除未使用的镜像

# 构建镜像
docker build -t myapp:1.0 .

3.2 容器操作

# 运行容器
docker run -d --name mynginx -p 80:80 nginx

# 查看运行中的容器
docker ps
docker container ls

# 查看所有容器(包括停止的)
docker ps -a

# 停止容器
docker stop mynginx

# 启动容器
docker start mynginx

# 重启容器
docker restart mynginx

# 删除容器
docker rm mynginx
docker container prune  # 删除停止的容器

# 查看容器日志
docker logs mynginx
docker logs -f mynginx  # 实时查看

# 进入容器
docker exec -it mynginx /bin/bash
docker exec -it mynginx sh

3.3 常用参数说明

参数说明示例
-d后台运行docker run -d nginx
-p端口映射-p 8080:80
-v挂载卷-v /data:/app/data
-e环境变量-e MYSQL_ROOT_PASSWORD=123456
--name容器名称--name myapp
--restart重启策略--restart=always

四、Dockerfile实战

4.1 Dockerfile基础语法

# FROM: 指定基础镜像
FROM python:3.10-slim

# LABEL: 添加元数据
LABEL maintainer="your-email@example.com"
LABEL version="1.0"

# WORKDIR: 设置工作目录
WORKDIR /app

# COPY: 复制文件到容器
COPY requirements.txt .

# RUN: 执行命令
RUN pip install --no-cache-dir -r requirements.txt

# COPY: 复制应用代码
COPY . .

# EXPOSE: 声明暴露的端口
EXPOSE 8000

# ENV: 设置环境变量
ENV PYTHONUNBUFFERED=1

# CMD: 容器启动时执行的命令
CMD ["python", "app.py"]

4.2 实战案例1:Python Flask应用

项目结构:

flask-app/
├── app.py
├── requirements.txt
├── Dockerfile
└── .dockerignore

app.py:

from flask import Flask, jsonify
app = Flask(__name__)
@app.route('/')
def hello():
    return jsonify({
        "message": "Hello from Docker!",
        "status": "success"
    })
@app.route('/health')
def health():
    return jsonify({"status": "healthy"})
if __name__ == '__main__':
    app.run(host='0.0.0.0', port=8000)

requirements.txt:

flask==3.0.0
gunicorn==21.2.0

Dockerfile:

FROM python:3.10-slim
LABEL maintainer="your-email@example.com"
WORKDIR /app
COPY requirements.txt .
RUN pip install --no-cache-dir -r requirements.txt
COPY . .
EXPOSE 8000
CMD ["gunicorn", "-w", "4", "-b", "0.0.0.0:8000", "app:app"]

.dockerignore:

__pycache__
*.pyc
.git
.gitignore
venv

构建和运行:

# 构建镜像
docker build -t flask-app:1.0 .

# 运行容器
docker run -d --name flask-app -p 8000:8000 flask-app:1.0

# 测试
curl http://localhost:8000

4.3 实战案例2:Node.js应用

项目结构:

node-app/
├── package.json
├── index.js
├── Dockerfile
└── .dockerignore

package.json:

{
  "name": "node-docker-app",
  "version": "1.0.0",
  "main": "index.js",
  "scripts": {
    "start": "node index.js"
  },
  "dependencies": {
    "express": "^4.18.2"
  }
}

index.js:

const express = require('express');
const app = express();
const PORT = process.env.PORT || 3000;
app.get('/', (req, res) => {
    res.json({
        message: 'Hello from Node.js in Docker!',
        timestamp: new Date().toISOString()
    });
});
app.get('/api/users', (req, res) => {
    res.json([
        { id: 1, name: 'Alice' },
        { id: 2, name: 'Bob' }
    ]);
});
app.listen(PORT, () => {
    console.log(`Server running on port ${PORT}`);
});

Dockerfile(多阶段构建):

# 阶段1: 构建阶段
FROM node:18-alpine AS builder
WORKDIR /app
COPY package*.json ./
RUN npm ci --only=production
COPY . .
# 阶段2: 运行阶段
FROM node:18-alpine
WORKDIR /app
COPY --from=builder /app/node_modules ./node_modules
COPY --from=builder /app/package*.json ./
COPY --from=builder /app/index.js ./
EXPOSE 3000
CMD ["node", "index.js"]

构建和运行:

# 构建镜像
docker build -t node-app:1.0 .

# 运行容器
docker run -d --name node-app -p 3000:3000 node-app:1.0

# 测试
curl http://localhost:3000

4.4 实战案例3:Nginx静态网站

项目结构:

nginx-site/
├── index.html
├── css/
│   └── style.css
├── js/
│   └── app.js
└── nginx.conf

index.html:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Docker Nginx Demo</title>
    <link rel="stylesheet" href="css/style.css" rel="external nofollow" >
</head>
<body>
    <div class="container">
        <h1>🐳 Welcome to Docker Nginx</h1>
        <p>This site is served by Nginx in a Docker container!</p>
        <button id="btn">Click Me</button>
    </div>
    <script src="js/app.js"></script>
</body>
</html>

nginx.conf:

events {
    worker_connections 1024;
}
http {
    server {
        listen 80;
        server_name localhost;
        root /usr/share/nginx/html;
        index index.html;
        location / {
            try_files $uri $uri/ /index.html;
        }
        location /api/ {
            proxy_pass http://backend:8000;
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
        }
    }
}

Dockerfile:

FROM nginx:alpine
LABEL maintainer="your-email@example.com"
COPY nginx.conf /etc/nginx/nginx.conf
COPY . /usr/share/nginx/html
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]

五、Docker Compose实战

5.1 Docker Compose简介

Docker Compose是用于定义和运行多容器Docker应用程序的工具。通过YAML文件配置应用的服务,然后使用一个命令创建和启动所有服务。

5.2 docker-compose.yml基础语法

version: '3.8'
services:
  web:
    image: nginx:alpine
    ports:
      - "80:80"
    volumes:
      - ./html:/usr/share/nginx/html
    depends_on:
      - app
  app:
    build: ./app
    ports:
      - "8000:8000"
    environment:
      - DATABASE_URL=postgres://user:pass@db:5432/mydb
    depends_on:
      - db
  db:
    image: postgres:15
    environment:
      - POSTGRES_USER=user
      - POSTGRES_PASSWORD=pass
      - POSTGRES_DB=mydb
    volumes:
      - postgres_data:/var/lib/postgresql/data
volumes:
  postgres_data:

5.3 实战案例:完整Web应用栈

项目结构:

fullstack-app/
├── docker-compose.yml
├── frontend/
│   ├── Dockerfile
│   └── src/
├── backend/
│   ├── Dockerfile
│   ├── requirements.txt
│   └── app.py
└── db/
    └── init.sql

docker-compose.yml:

version: '3.8'
services:
  # 前端服务
  frontend:
    build:
      context: ./frontend
      dockerfile: Dockerfile
    ports:
      - "80:80"
    depends_on:
      - backend
    networks:
      - app-network
  # 后端服务
  backend:
    build:
      context: ./backend
      dockerfile: Dockerfile
    ports:
      - "8000:8000"
    environment:
      - DATABASE_HOST=db
      - DATABASE_PORT=5432
      - DATABASE_NAME=appdb
      - DATABASE_USER=appuser
      - DATABASE_PASSWORD=apppass
    depends_on:
      - db
    networks:
      - app-network
    volumes:
      - ./backend:/app
  # 数据库服务
  db:
    image: postgres:15-alpine
    environment:
      - POSTGRES_DB=appdb
      - POSTGRES_USER=appuser
      - POSTGRES_PASSWORD=apppass
    volumes:
      - postgres_data:/var/lib/postgresql/data
      - ./db/init.sql:/docker-entrypoint-initdb.d/init.sql
    ports:
      - "5432:5432"
    networks:
      - app-network
  # Redis缓存
  redis:
    image: redis:7-alpine
    ports:
      - "6379:6379"
    networks:
      - app-network
volumes:
  postgres_data:
networks:
  app-network:
    driver: bridge

backend/app.py:

from flask import Flask, jsonify
import psycopg2
from redis import Redis
import os
app = Flask(__name__)
def get_db_connection():
    conn = psycopg2.connect(
        host=os.getenv('DATABASE_HOST', 'db'),
        port=int(os.getenv('DATABASE_PORT', 5432)),
        database=os.getenv('DATABASE_NAME', 'appdb'),
        user=os.getenv('DATABASE_USER', 'appuser'),
        password=os.getenv('DATABASE_PASSWORD', 'apppass')
    )
    return conn
def get_redis():
    return Redis(
        host='redis',
        port=6379,
        decode_responses=True
    )
@app.route('/')
def index():
    return jsonify({
        "message": "Fullstack API",
        "services": {
            "database": "PostgreSQL",
            "cache": "Redis"
        }
    })
@app.route('/health')
def health():
    try:
        redis = get_redis()
        redis.ping()
        conn = get_db_connection()
        cursor = conn.cursor()
        cursor.execute('SELECT 1')
        cursor.close()
        conn.close()
        return jsonify({
            "status": "healthy",
            "database": "connected",
            "redis": "connected"
        })
    except Exception as e:
        return jsonify({
            "status": "unhealthy",
            "error": str(e)
        }), 500
if __name__ == '__main__':
    app.run(host='0.0.0.0', port=8000)

backend/Dockerfile:

FROM python:3.10-slim
WORKDIR /app
COPY requirements.txt .
RUN pip install --no-cache-dir -r requirements.txt
COPY . .
EXPOSE 8000
CMD ["python", "app.py"]

backend/requirements.txt:

flask==3.0.0
psycopg2-binary==2.9.9
redis==5.0.1

db/init.sql:

CREATE TABLE IF NOT EXISTS users (
    id SERIAL PRIMARY KEY,
    name VARCHAR(100) NOT NULL,
    email VARCHAR(100) UNIQUE NOT NULL,
    created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
);
INSERT INTO users (name, email) VALUES
    ('Alice', 'alice@example.com'),
    ('Bob', 'bob@example.com');

启动应用:

# 构建并启动所有服务
docker-compose up -d

# 查看服务状态
docker-compose ps

# 查看日志
docker-compose logs -f

# 停止服务
docker-compose down

# 停止并删除卷
docker-compose down -v

六、数据持久化

6.1 数据卷(Volume)

# 创建数据卷
docker volume create mydata

# 查看数据卷
docker volume ls
docker volume inspect mydata

# 使用数据卷
docker run -d --name myapp -v mydata:/app/data nginx

# 删除数据卷
docker volume rm mydata

6.2 绑定挂载(Bind Mount)

# 挂载本地目录
docker run -d --name myapp \
    -v /host/path:/container/path \
    nginx

# 只读挂载
docker run -d --name myapp \
    -v /host/path:/container/path:ro \
    nginx

6.3 实战案例:MySQL数据持久化

version: '3.8'
services:
  mysql:
    image: mysql:8.0
    environment:
      MYSQL_ROOT_PASSWORD: rootpass
      MYSQL_DATABASE: myapp
      MYSQL_USER: appuser
      MYSQL_PASSWORD: apppass
    volumes:
      # 数据持久化
      - mysql_data:/var/lib/mysql
      # 初始化脚本
      - ./init.sql:/docker-entrypoint-initdb.d/init.sql
      # 配置文件
      - ./my.cnf:/etc/mysql/conf.d/custom.cnf:ro
    ports:
      - "3306:3306"
volumes:
  mysql_data:

七、网络管理

7.1 Docker网络类型

网络类型说明使用场景
bridge默认网络,容器间通过桥接通信单机多容器通信
host容器使用宿主机网络需要高性能网络
none无网络完全隔离
overlay跨主机网络Docker Swarm集群

7.2 创建自定义网络

# 创建bridge网络
docker network create mynetwork

# 查看网络
docker network ls
docker network inspect mynetwork

# 连接容器到网络
docker network connect mynetwork mycontainer

# 断开网络
docker network disconnect mynetwork mycontainer

# 删除网络
docker network rm mynetwork

7.3 实战案例:多服务网络通信

version: '3.8'
services:
  web:
    image: nginx:alpine
    ports:
      - "80:80"
    networks:
      - frontend
      - backend
  app:
    build: ./app
    networks:
      - backend
  db:
    image: postgres:15
    networks:
      - backend
networks:
  frontend:
    driver: bridge
  backend:
    driver: bridge

八、Docker最佳实践

8.1 镜像优化

1. 使用多阶段构建

# 构建阶段
FROM node:18 AS builder
WORKDIR /app
COPY package*.json ./
RUN npm ci
COPY . .
RUN npm run build

# 运行阶段
FROM nginx:alpine
COPY --from=builder /app/dist /usr/share/nginx/html

2. 减少层数

# 不好的做法
RUN apt-get update
RUN apt-get install -y python3
RUN apt-get install -y pip

# 好的做法
RUN apt-get update && \
    apt-get install -y python3 pip && \
    apt-get clean && \
    rm -rf /var/lib/apt/lists/*

3. 使用.dockerignore

.git
.gitignore
node_modules
__pycache__
*.log
.env

8.2 安全最佳实践

# 使用非root用户
FROM python:3.10-slim

WORKDIR /app

COPY requirements.txt .
RUN pip install --no-cache-dir -r requirements.txt

COPY . .

# 创建非root用户
RUN useradd -m appuser && \
    chown -R appuser:appuser /app

USER appuser

EXPOSE 8000

CMD ["python", "app.py"]

8.3 健康检查

version: '3.8'
services:
  web:
    image: nginx:alpine
    ports:
      - "80:80"
    healthcheck:
      test: ["CMD", "wget", "--quiet", "--tries=1", "--spider", "http://localhost/health"]
      interval: 30s
      timeout: 10s
      retries: 3
      start_period: 40s
  app:
    build: ./app
    healthcheck:
      test: ["CMD", "curl", "-f", "http://localhost:8000/health"]
      interval: 30s
      timeout: 10s
      retries: 3

8.4 日志管理

version: '3.8'
services:
  app:
    image: myapp:1.0
    logging:
      driver: "json-file"
      options:
        max-size: "10m"
        max-file: "3"

九、常见问题与解决方案

Q1: 容器启动失败怎么办?

排查步骤:

# 查看容器日志
docker logs <container_id>

# 查看容器详细信息
docker inspect <container_id>

# 进入容器调试
docker exec -it <container_id> /bin/bash

# 查看容器状态
docker ps -a

Q2: 如何清理Docker资源?

# 清理停止的容器
docker container prune

# 清理未使用的镜像
docker image prune -a

# 清理未使用的卷
docker volume prune

# 清理未使用的网络
docker network prune

# 清理所有未使用的资源
docker system prune -a --volumes

Q3: 如何在容器和宿主机之间复制文件?

# 从宿主机复制到容器
docker cp /host/file.txt container:/container/path/

# 从容器复制到宿主机
docker cp container:/container/file.txt /host/path/

Q4: 如何限制容器资源?

version: '3.8'
services:
  app:
    image: myapp:1.0
    deploy:
      resources:
        limits:
          cpus: '0.50'
          memory: 512M
        reservations:
          cpus: '0.25'
          memory: 256M

十、完整实战项目

10.1 项目概述

构建一个完整的博客系统,包含:

10.2 项目结构

blog-system/
├── docker-compose.yml
├── frontend/
│   ├── Dockerfile
│   └── src/
├── backend/
│   ├── Dockerfile
│   ├── requirements.txt
│   └── app.py
├── nginx/
│   └── nginx.conf
└── db/
    └── init.sql

10.3 docker-compose.yml

version: '3.8'
services:
  # 前端服务
  frontend:
    build:
      context: ./frontend
      dockerfile: Dockerfile
    ports:
      - "3000:3000"
    depends_on:
      - backend
    networks:
      - blog-network
  # 后端服务
  backend:
    build:
      context: ./backend
      dockerfile: Dockerfile
    ports:
      - "8000:8000"
    environment:
      - DATABASE_HOST=db
      - DATABASE_PORT=5432
      - DATABASE_NAME=blogdb
      - DATABASE_USER=bloguser
      - DATABASE_PASSWORD=blogpass
      - REDIS_HOST=redis
      - REDIS_PORT=6379
    depends_on:
      - db
      - redis
    networks:
      - blog-network
    volumes:
      - ./backend:/app
  # Nginx反向代理
  nginx:
    image: nginx:alpine
    ports:
      - "80:80"
    volumes:
      - ./nginx/nginx.conf:/etc/nginx/nginx.conf:ro
    depends_on:
      - frontend
      - backend
    networks:
      - blog-network
  # PostgreSQL数据库
  db:
    image: postgres:15-alpine
    environment:
      - POSTGRES_DB=blogdb
      - POSTGRES_USER=bloguser
      - POSTGRES_PASSWORD=blogpass
    volumes:
      - postgres_data:/var/lib/postgresql/data
      - ./db/init.sql:/docker-entrypoint-initdb.d/init.sql
    networks:
      - blog-network
  # Redis缓存
  redis:
    image: redis:7-alpine
    ports:
      - "6379:6379"
    networks:
      - blog-network
volumes:
  postgres_data:
networks:
  blog-network:
    driver: bridge

10.4 启动项目

# 克隆项目
git clone <repository-url>
cd blog-system

# 构建并启动所有服务
docker-compose up -d --build

# 查看服务状态
docker-compose ps

# 查看日志
docker-compose logs -f

# 访问应用
# 前端: http://localhost:3000
# 后端: http://localhost:8000
# Nginx: http://localhost

十一、总结

本文从Docker的基础概念出发,通过多个实战案例详细介绍了容器化技术的应用:

通过本文的学习,你应该能够:

记住:Docker的核心价值在于"一次构建,到处运行",掌握它将极大提升你的开发效率!

以上就是从入门到精通详解Docker容器化的实战指南的详细内容,更多关于Docker容器化的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文