20-项目部署(Docker)¶
在昨天的课程中,我们学习了 Linux 操作系统的常见命令,在 Linux 上安装软件,以及如何在 Linux 上部署一个单体项目。大家想一想自己最大的感受是什么?
我相信,除了个别天赋异禀的同学以外,大多数同学都会有相同的感受,那就是麻烦。核心体现在三点:
- 命令太多了,记不住
- 软件安装包名字复杂,不知道去哪里找
- 安装和部署步骤复杂,容易出错
其实上述问题不仅仅是新手,即便是运维在安装、部署的时候一样会觉得麻烦、容易出错 。
特别是我们即将进入项目阶段的学习,项目动辄就是几台、几十台、甚至上百台服务需要部署,有些大型项目甚至达到成千上万台服务。而由于每台服务器的运行环境不同,你写好的安装流程、部署脚本并不一定在每个服务器都能正常运行,经常会出错。这就给系统的部署运维带来了很多困难 。
那么,有没有一种技术能够避免部署对服务器环境的依赖,减少复杂的部署流程呢?
答案是肯定的,这就是我们今天要学习的 Docker 技术。你会发现,有了 Docker 以后项目的部署如丝般顺滑,大大减少了运维工作量。
即便你对 Linux 不熟悉,你也能轻松部署各种常见软件、Java 项目。
[!TIP] 通过今天的学习,大家要能够达成下面的学习目标:
- 能利用 Docker 部署常见软件
- 能利用 Docker 打包并部署 Java 应用
- 理解 Docker 数据卷的基本作用
- 能看懂 DockerCompose 文件
快速入门¶
要想让 Docker 帮我们安装和部署软件,肯定要保证你的机器上有 Docker。由于大家的操作系统各不相同,安装方式也不同。为了便于大家学习,我们统一在之前提供给大家的 CentOS 的虚拟机中已经安装了 Docker,统一学习环境。
如果大家需要自己在别的机器上安装 Docker 环境,可以参照最后的 附录中的:Docker 安装文档
部署 MySQL¶
首先,我们利用 Docker 来安装一个 MySQL 软件,大家可以对比一下之前传统的安装方式,看看哪个效率更高一些。
如果是利用传统方式部署 MySQL,大概的步骤有:
- 搜索并下载 MySQL 安装包
- 上传至 Linux 环境
- 解压和配置环境
- 安装
- 初始化和配置
而使用 Docker 安装,仅仅需要一步即可,在命令行输入下面的命令(建议采用 CV 大法):
docker run -d \
--name mysql \
-p 3307:3306 \
-e TZ=Asia/Shanghai \
-e MYSQL_ROOT_PASSWORD=123 \
mysql:8
运行效果如图(在给大家提供的资料中,已经下载好了 mysql 8 版本的镜像):

MySQL 安装完毕!通过任意客户端工具即可连接到 MySQL。
说明:
其实,当我们执行命令后,Docker 做的第一件事情,是去自动搜索并下载了 MySQL,然后会自动运行 MySQL,我们完全不用插手,是非常方便的 。
而且,这种安装方式你完全不用考虑运行的操作系统环境,它不仅仅在 CentOS 系统是这样,在 Ubuntu 系统、macOS 系统、甚至是装了 WSL 的 Windows 下,都可以使用这条命令来安装 MySQL 。
要知道,不同操作系统下其安装包、运行环境是都不相同的!如果是手动安装,必须手动解决安装包不同、环境不同的、配置不同的问题!
而使用 Docker,这些完全不用考虑。就是因为 Docker 会自动搜索并下载 MySQL。(甚至不用配防火墙)
注意:
这里下载的不是安装包,而是镜像。镜像中不仅包含了 MySQL 本身,还包含了其运行所需要的环境、配置、系统级函数库。因此它在运行时就有自己独立的环境,就可以跨系统运行,也不需要手动再次配置环境了。这套独立运行的隔离环境我们称为容器。
说明:
- 镜像:英文是 image
- 容器:英文是 container
[!TIP] 因此,Docker 安装软件的过程,就是自动搜索下载镜像,然后创建并运行容器的过程。
Docker 会根据命令中的镜像名称自动搜索并下载镜像,那么问题来了,它是去哪里搜索和下载镜像的呢?这些镜像又是谁制作的呢?
Docker 官方提供了一个专门管理、存储镜像的网站,并对外开放了镜像上传、下载的权利。Docker 官方提供了一些基础镜像,然后各大软件公司又在基础镜像基础上,制作了自家软件的镜像,全部都存放在这个网站。这个网站就成了 Docker 镜像交流的社区:https://hub.docker.com
基本上我们常用的各种软件都能在这个网站上找到,我们甚至可以自己制作镜像上传上去。【但是该网站,目前国内上不去了】
像这种提供存储、管理 Docker 镜像的服务器,被称为 DockerRegistry,可以翻译为镜像仓库。DockerHub 网站是官方仓库,阿里云、华为云会提供一些第三方仓库,我们也可以自己搭建私有的镜像仓库。
官方仓库在国外,下载速度较慢,一般我们都会使用第三方仓库提供的镜像加速功能,提高下载速度。而企业内部的机密项目,往往会采用私有镜像仓库。
总之,镜像的来源有两种:
- 基于官方基础镜像自己制作
- 直接去 DockerRegistry 下载
[!TIP] 总结一下: Docker 本身包含一个后台服务,我们可以利用 Docker 命令告诉 Docker 服务,帮助我们快速部署指定的应用。Docker 服务部署应用时,首先要去搜索并下载应用对应的镜像,然后根据镜像创建并允许容器,应用就部署完成了。
用一幅图表示如下:
如果没有docker,似乎是很难做到同时运行多个mysql的.
命令解读¶
利用 Docker 快速的安装了 MySQL,非常的方便,不过我们执行的命令到底是什么意思呢?
docker run -d \
--name mysql \
-p 3307:3306 \
-e TZ=Asia/Shanghai \
-e MYSQL_ROOT_PASSWORD=123 \
mysql:8
[!TIP] 解读:
docker run -d:创建并运行一个容器,-d则是让容器以后台进程运行(ps:输出还是在前台)--name mysql: 给容器起个名字叫mysql,你可以叫别的(容器的名字必须是唯一的)-p 3307:3306: 设置端口映射。- 容器是隔离环境,外界不可访问。但是可以将宿主机端口映射容器内到端口,当访问宿主机指定端口时,就是在访问容器内的端口了。
- 容器内端口往往是由容器内的进程决定,例如 MySQL 进程默认端口是 3306,因此容器内端口一定是 3306;而宿主机端口则可以任意指定,一般与容器内保持一致。
- 格式:
-p 宿主机端口:容器内端口,示例中就是将宿主机的 3307 映射到容器内的 3306 端口ps:其实docker的容器也有ip,并且和宿主机之间组成了nat
-``e`` TZ=Asia/Shanghai: 配置容器内进程运行时的一些参数- 格式:
-e KEY=VALUE,KEY 和 VALUE 都由容器内进程决定案例中,
TZ=Asia/Shanghai是设置时区;MYSQL_ROOT_PASSWORD=123是设置 MySQL 默认密码
mysql:8: 设置镜像名称,Docker 会根据这个名字搜索并下载镜像- 格式:
REPOSITORY:TAG,例如mysql:8.0,其中REPOSITORY可以理解为镜像名,TAG是版本号- 在未指定
TAG的情况下,默认是最新版本,也就是mysql:latest
镜像的名称不是随意的,而是要到 DockerRegistry 中寻找,镜像运行时的配置也不是随意的,要参考镜像的帮助文档,这些在 DockerHub 网站或者软件的官方网站中都能找到。
如果我们要安装其它软件,也可以到 DockerRegistry 中寻找对应的镜像名称和版本,阅读相关配置即可。
docker run 命令中的 -e 或 --env 选项用于设置容器内部运行的进程的环境变量。 简而言之,它指定的是容器的环境变量,而不是宿主机(Host)的环境变量。
Docker 基础¶
接下来,我们一起来学习 Docker 使用的一些基础知识,为将来部署项目打下基础。
常见命令¶
命令介绍¶
其中,比较常见的命令有:
| **命令** | **说明** | **文档地址** |
| docker pull | 拉取镜像 | [docker pull](https://docs.docker.com/engine/reference/commandline/pull/) |
| docker push | 推送镜像到DockerRegistry | [docker push](https://docs.docker.com/engine/reference/commandline/push/) |
| docker images | 查看本地镜像 | [docker images](https://docs.docker.com/engine/reference/commandline/images/) |
| docker rmi | 删除本地镜像 | [docker rmi](https://docs.docker.com/engine/reference/commandline/rmi/) |
| docker run | 创建并运行容器(不能重复创建) | [docker run](https://docs.docker.com/engine/reference/commandline/run/) |
| docker stop | 停止指定容器 | [docker stop](https://docs.docker.com/engine/reference/commandline/stop/) |
| docker start | 启动指定容器 | [docker start](https://docs.docker.com/engine/reference/commandline/start/) |
| docker restart | 重新启动容器 | [docker restart](https://docs.docker.com/engine/reference/commandline/restart/) |
| docker rm | 删除指定容器 | [docs.docker.com](https://docs.docker.com/engine/reference/commandline/rm/) |
| docker ps | 查看容器 | [docker ps](https://docs.docker.com/engine/reference/commandline/ps/) |
| docker logs | 查看容器运行日志 | [docker logs](https://docs.docker.com/engine/reference/commandline/logs/) |
| docker exec | 进入容器 | [docker exec](https://docs.docker.com/engine/reference/commandline/exec/) |
| docker save | 保存镜像到本地压缩文件 | [docker save](https://docs.docker.com/engine/reference/commandline/save/) |
| docker load | 加载本地压缩文件到镜像 | [docker load](https://docs.docker.com/engine/reference/commandline/load/) |
| docker inspect | 查看容器详细信息 | [docker inspect](https://docs.docker.com/engine/reference/commandline/inspect/) |
用一副图来表示这些命令的关系:(这个图不错)
其实就是类似于:镜像就是iso文件,容器就是虚拟机.虚拟机能停能启动,而且运行不影响iso.
另外,docker ps只能看到运行中的容器,而docker ps -a才是全部的.
补充:
默认情况下,每次重启虚拟机我们都需要手动启动 Docker 和 Docker 中的容器。通过命令可以实现开机自启:
# Docker开机自启
systemctl enable docker
# Docker容器开机自启
docker update --restart=always [容器名/容器id]
演示¶
我们以 Nginx 为例给大家演示上述命令。
# 第1步,去DockerHub查看nginx镜像仓库及相关信息
# 第2步,拉取Nginx镜像 (比较耗时)
docker pull nginx:1.20.2
# 第3步,查看镜像
docker images
# 第4步,创建并允许Nginx容器
docker run -d --name nginx -p 80:80 nginx
# 第5步,查看运行中容器
docker ps
# 也可以加格式化方式访问,格式会更加清爽
docker ps --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}"
# 第6步,访问网页,地址:http://虚拟机地址
# 第7步,停止容器
docker stop nginx
# 第8步,查看所有容器
docker ps -a --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}"
# 第9步,再次启动nginx容器
docker start nginx
# 第10步,再次查看容器
docker ps --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}"
# 第11步,查看容器详细信息
docker inspect nginx
# 第12步,进入容器,查看容器内目录
docker exec -it nginx bash//要求分配一个终端
# 或者,可以进入MySQL
docker exec -it mysql mysql -uroot -p
# 第13步,删除容器
docker rm nginx
# 发现无法删除,因为容器运行中,强制删除容器
docker rm -f nginx
docker容器里面是没有包括vim这样的东西的,因为要保持体积.
数据卷¶
容器是隔离环境,容器内程序的文件、配置、运行时产生的容器都在容器内部,我们要读写容器内的文件非常不方便。大家思考几个问题:
- 如果要升级 MySQL 版本,需要销毁旧容器,那么数据岂不是跟着被销毁了?
- MySQL、Nginx 容器运行后,如果我要修改其中的某些配置该怎么办?
- 我想要让 Nginx 代理我的静态资源怎么办?
因此,容器提供程序的运行环境,但是 程序运行产生的数据、程序运行依赖的配置都应该与容器解耦。
介绍¶
数据卷(volume)是一个虚拟目录,是容器内目录与宿主机目录之间映射的桥梁。
以 Nginx 为例,我们知道 Nginx 中有两个关键的目录:
html:放置一些静态资源conf:放置配置文件
如果我们要让 Nginx 代理我们的静态资源,最好是放到 html 目录;如果我们要修改 Nginx 的配置,最好是找到 conf 下的 nginx.conf 文件。
但遗憾的是,容器运行的 Nginx 所有的文件都在容器内部。所以我们必须利用数据卷将两个目录与宿主机目录关联,方便我们操作。
如图:

在上图中:
- 我们创建了两个数据卷:
conf、html - Nginx 容器内部的
conf目录和html目录分别与两个数据卷关联。 - 而数据卷 conf 和 html 分别指向了宿主机的
/var/lib/docker/volumes/conf/_data目录和/var/lib/docker/volumes/html/_data目录
这样以来,容器内的 conf 和 html 目录就 与宿主机的 conf 和 html 目录关联起来,我们称为挂载。
此时,我们操作宿主机的 /var/lib/docker/volumes/html/_data 就是在操作容器内的 /usr/share/nginx/html/``_data 目录。只要我们将静态资源放入宿主机对应目录,就可以被 Nginx 代理了。
[!TIP] 小提示:
/var/lib/docker/volumes这个目录就是默认的存放所有容器数据卷的目录,其下再根据数据卷名称创建新目录,格式为/数据卷名/_data。
为什么不让容器目录直接指向宿主机目录呢?
- 因为直接指向宿主机目录就与宿主机强耦合了,如果切换了环境,宿主机目录就可能发生改变了。由于容器一旦创建,目录挂载就无法修改,这样容器就无法正常工作了。
- 但是容器指向数据卷,一个逻辑名称,而数据卷再指向宿主机目录,就不存在强耦合。如果宿主机目录发生改变,只要改变数据卷与宿主机目录之间的映射关系即可。
不过,我们通过由于数据卷目录比较深,不好寻找,通常我们也允许让容器直接与宿主机目录挂载而不使用数据卷,具体参考 2.2.3 小节。
命令¶
数据卷的相关命令有:
| **命令** | **说明** | **文档地址** |
| docker volume create | 创建数据卷 | [docker volume create](https://docs.docker.com/engine/reference/commandline/volume_create/) |
| docker volume ls | 查看所有数据卷 | [docs.docker.com](https://docs.docker.com/engine/reference/commandline/volume_ls/) |
| docker volume rm | 删除指定数据卷 | [docs.docker.com](https://docs.docker.com/engine/reference/commandline/volume_prune/) |
| docker volume inspect | 查看某个数据卷的详情 | [docs.docker.com](https://docs.docker.com/engine/reference/commandline/volume_inspect/) |
| docker volume prune | 清除数据卷 | [docker volume prune](https://docs.docker.com/engine/reference/commandline/volume_prune/) |
注意:容器与数据卷的挂载要在创建容器时配置,对于创建好的容器,是不能设置数据卷的。而且创建容器的过程中,数据卷会自动创建。
教学演示环节:演示一下 nginx 的 html 目录挂载
# 1.首先创建容器并指定数据卷,注意通过 -v 参数来指定数据卷
docker run -d --name nginx -p 80:80 -v html:/usr/share/nginx/html nginx:1.20.2
# 2.然后查看数据卷
docker volume ls
# DRIVER VOLUME NAME
local 007e5627b3baed90169b8fc47cf2fe8d31e4329310665ae89a8ac30334c38ea8
local
d5e99aa3d3435d929fddd239e42587a6c523bf8a6e5e444ec09c8b4fa1648aac
local
html
前两个是mysql自动创建匿名数据卷(由于有两个mysql容器,因此有两个),后面再说
# 3.查看数据卷详情
docker volume inspect html
# 4.查看/var/lib/docker/volumes/html/_data目录
ll /var/lib/docker/volumes/html/_data
这样就会显示docker里的/usr/share/nginx/html的内容.
# 5.进入该目录,并随意修改index.html内容
cd /var/lib/docker/volumes/html/_data
vi index.html
# 6.打开页面,查看效果
# 7.进入容器内部,查看/usr/share/nginx/html目录内的文件是否变化
docker exec -it nginx bash
教学演示环节:演示一下 MySQL 的匿名数据卷
# 1.查看MySQL容器详细信息
docker inspect mysql
# 关注其中.Config.Volumes部分和.Mounts部分
我们关注两部分内容,第一是 .Config.Volumes 部分:
{
"Config": {
// ... 略
"Volumes": {
"/var/lib/mysql": {}
}
// ... 略
}
}
可以发现这个容器声明了一个本地目录(docker内的),需要挂载数据卷,但是数据卷未定义。这就是匿名卷。
然后,我们再看结果中的 .Mounts 部分:
{
"Mounts": [
{
"Type": "volume",
"Name": "29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f",
"Source": "/var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data",
"Destination": "/var/lib/mysql",
"Driver": "local",
}
]
}
可以发现,其中有几个关键属性:
- Name:数据卷名称。由于定义容器未设置容器名,这里的就是匿名卷自动生成的名字,一串 hash 值。
- Source:宿主机目录
- Destination : 容器内的目录
上述配置是将容器内的 /var/lib/mysql 这个目录,与数据卷 29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f 挂载。于是在宿主机中就有了 /var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data 这个目录。这就是匿名数据卷对应的目录,其使用方式与普通数据卷没有差别。
接下来,可以查看该目录下的 MySQL 的 data 文件:
ls -l /var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data
容器被删除后,曾经挂载的数据卷是还是会留存的.这样,实际上是实现了数据备份
注意:每一个不同的镜像,将来创建容器后内部有哪些目录可以挂载,可以参考 DockerHub 对应的页面 。
挂载本地目录或文件¶
可以发现,数据卷的目录结构较深,如果我们去操作数据卷目录会不太方便。在很多情况下,我们会直接将容器目录与宿主机指定目录挂载。挂载语法与数据卷类似:
# 挂载本地目录
-v 本地目录:容器内目录
# 挂载本地文件
-v 本地文件:容器内文件
注意:本地目录或文件必须以 / 或 ./ 开头,如果直接以名字开头,会被识别为数据卷名而非本地目录名。
例如:
-v mysql:/var/lib/mysql # 会被识别为一个数据卷叫mysql,运行时会自动创建这个数据卷
-v ./mysql:/var/lib/mysql # 会被识别为当前目录下的mysql目录,运行时如果不存在会创建目录
教学演示,删除并重新创建 mysql 容器,并完成本地目录挂载:
- 挂载
/root/mysql/data到容器内的/var/lib/mysql目录(数据目录) - 挂载
/root/mysql/init到容器内的/docker-entrypoint-initdb.d目录(初始化的 SQL 脚本目录,在容器启动的时候会运行一次里面的脚本.) - 挂载
/root/mysql/conf到容器内的/etc/mysql/conf.d目录(这个是 MySQL 配置文件目录)
在课前资料中已经准备好了 mysql 的 init 目录、conf 目录、data 目录,可以直接将其上传到 Linux 服务器中的 /root/mysql 目录下。
最终执行的指令如下:
docker run -d \
--name mysql \
-p 3307:3306 \
-e MYSQL_ROOT_PASSWORD=123 \
-e TZ=Asia/Shanghai \
-v /root/mysql/data:/var/lib/mysql \
-v /root/mysql/init:/docker-entrypoint-initdb.d \
-v /root/mysql/conf:/etc/mysql/conf.d \
mysql:8
自定义镜像¶
前面我们一直在使用别人准备好的镜像,那如果我要部署一个 Java 项目,把它打包为一个镜像该怎么做呢? 那接下来,我们就来介绍一下如何自定义镜像。
镜像结构¶
要想自己构建镜像,必须先了解镜像的结构。
之前我们说过,镜像之所以能让我们快速跨操作系统部署应用而忽略其运行环境、配置,就是因为镜像中包含了程序运行需要的系统函数库、环境、配置、依赖。
因此,自定义镜像本质就是依次准备好程序运行的基础环境、依赖、应用本身、运行配置等文件,并且打包而成。
举个例子,我们要从 0 部署一个 Java 应用,大概流程是这样:
- 准备一个 linux 服务(CentOS 或者 Ubuntu 均可)
- 安装并配置 JDK
- 上传 Jar 包
- 运行 jar 包
那因此,我们打包镜像也是分成这么几步:
- 准备 Linux 运行环境(java 项目并不需要完整的操作系统,仅仅是基础运行环境即可)
- 安装并配置 JDK
- 拷贝 jar 包
- 配置启动脚本
上述步骤中的每一次操作其实都是在生产一些文件(系统运行环境、函数库、配置最终都是磁盘文件),所以镜像就是一堆文件的集合。
但需要注意的是,镜像文件不是随意堆放的,而是按照操作的步骤分层叠加而成,每一层形成的文件都会单独打包并标记一个唯一 id,称为 Layer(层)。这样,如果我们构建时用到的某些层其他人已经制作过,就可以直接拷贝使用这些层,而不用重复制作。(我们pull的时候会看到输出的一堆东西的id就是一个的层)
例如,第一步中需要的 Linux 运行环境,通用性就很强,所以 Docker 官方就制作了这样的只包含 Linux 运行环境的镜像。我们在制作 java 镜像时,就无需重复制作,直接使用 Docker 官方提供的 CentOS 或 Ubuntu 镜像作为基础镜像。然后再搭建其它层即可,这样逐层搭建,最终整个 Java 项目的镜像结构如图所示:

Dockerfile¶
由于制作镜像的过程中,需要逐层处理和打包,比较复杂,所以 Docker 就提供了自动打包镜像的功能。我们只需要将打包的过程,每一层要做的事情用固定的语法写下来,交给 Docker 去执行即可。而这种记录镜像结构的文件就称为 Dockerfile。注意,这个名字是固定的,不能修改
其中的语法比较多,比较常用的有:
| **指令** | **说明** | **示例** |
| **FROM** | 指定基础镜像 | `FROM centos:7` |
| **ENV** | 设置环境变量,可在后面指令使用 | `ENV key value` |
| **COPY** | 拷贝本地文件到镜像的指定目录 | `COPY ./xx.jar /tmp/app.jar` |
| **RUN** | 执行Linux的shell命令,一般是安装过程的命令 | `RUN yum install gcc` |
| **EXPOSE** | 指定容器运行时监听的端口,是给镜像使用者看的 | EXPOSE 8080 |
| **ENTRYPOINT** | 镜像中应用的启动命令,容器运行时调用 | ENTRYPOINT java -jar xx.jar |
例如,要基于 centos:7 镜像来构建一个 Java 应用,其 Dockerfile 内容如下:
# 使用 CentOS 7 作为基础镜像
FROM centos:7
# 添加 JDK 到镜像中
COPY jdk17.tar.gz /usr/local/
RUN tar -xzf /usr/local/jdk17.tar.gz -C /usr/local/ && rm /usr/local/jdk17.tar.gz
//Ps:-C选项是用来指定解压后放到哪里
# 设置环境变量(这个环境变量是容器内部的环境变量)
ENV JAVA_HOME=/usr/local/jdk-17.0.10
ENV PATH=$JAVA_HOME/bin:$PATH
# 创建应用目录
RUN mkdir -p /app
WORKDIR /app //其实就是类似于cd,切换到某个目录下
# 复制应用 JAR 文件到容器
COPY app.jar app.jar
# 暴露端口
EXPOSE 8080
# 运行命令
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/app/app.jar"]
Dockerfile 文件编写好了之后,就可以使用如下命令来构建镜像了。
docker build -t 镜像名 .
- -t :是给镜像起名,格式依然是 repository:tag 的格式,不指定 tag 时,默认为 latest
- . :是指定 Dockerfile 所在目录,如果就在当前目录,则指定为"."
演示:

网络¶
上节课我们创建了一个 Java 项目的容器,而 Java 项目往往需要访问其它各种中间件,例如 MySQL、Redis 等。现在,我们的容器之间能否互相访问呢?我们来测试一下
首先,我们查看下 MySQL 容器的详细信息,重点关注其中的网络 IP 地址:
# 1.用基本命令,寻找Networks.bridge.IPAddress属性
docker inspect mysql
# 也可以使用format过滤结果
docker inspect --format='{{range .NetworkSettings.Networks}}{{println .IPAddress}}{{end}}' mysql
# 得到IP地址如下:
172.17.0.2
# 2.然后通过命令进入dd容器
docker exec -it dd bash
# 3.在容器内,通过ping命令测试网络
ping 172.17.0.2
_# 结果_
_PING 172.17.0.2 (172.17.0.2) 56(84) bytes of data._
_64 bytes from 172.17.0.2: icmp_seq=1 ttl=64 time=0.053 ms_
_64 bytes from 172.17.0.2: icmp_seq=2 ttl=64 time=0.059 ms_
_64 bytes from 172.17.0.2: icmp_seq=3 ttl=64 time=0.058 ms_
发现可以互联,没有问题。也就是说,默认情况下.所有的容器都是在同一个网段/同一个局域网里的. 实际上,默认情况下
我们也可以通过ifconfig来看.其中docker0这个网卡就是.
但是,容器的网络 IP 其实是一个虚拟的 IP,其值并不固定与某一个容器绑定,如果我们在开发时写死某个 IP,而在部署时很可能 MySQL 容器的 IP 会发生变化,连接会失败。也就是说,关闭一个容器再启动,ip可能会变.
常见命令有:
| **命令** | **说明** |
| docker network create | 创建一个网络 |
| docker network ls | 查看所有网络 |
| docker network rm | 删除指定网络 |
| docker network prune | 清除未使用的网络 |
| docker network connect | 使指定容器连接加入某网络 |
| docker network disconnect | 使指定容器连接离开某网络 |
| docker network inspect | 查看网络详细信息 |
教学演示:自定义网络
# 1.首先通过命令创建一个网络
docker network create itheima
# 2.然后查看网络
docker network ls
# 结果:
_NETWORK ID NAME DRIVER SCOPE_
_639bc44d0a87 bridge bridge local_
_403f16ec62a2 itheima bridge local_
_0dc0f72a0fbb host host local_
_cd8d3e8df47b none null local_
# 其中,除了itheima以外,其它都是默认的网络
# 3.让 myapp 和 mysql 都加入该网络
# 3.1.mysql容器,加入 itheima 网络
docker network connect itheima mysql
# 3.2.myapp容器,也就是我们的java项目, 加入 itheima 网络
docker network connect itheima myapp
这种情况下,这两个容器相当于都有bridge和itheima这两张网卡.
# 4.进入dd容器,尝试利用别名访问db
# 4.1.进入容器
docker exec -it myapp bash
# 4.2.用容器名访问
ping mysql
# 结果:
_PING mysql (172.18.0.2) 56(84) bytes of data._
_64 bytes from mysql.itheima (172.18.0.2): icmp_seq=1 ttl=64 time=0.044 ms_
_64 bytes from mysql.itheima (172.18.0.2): icmp_seq=2 ttl=64 time=0.054 ms_
另外也可以在创建容器的时候直接指定:
这种情况下就只有ithema这一张网卡.
项目部署¶
部署服务端¶
- 需求:将我们开发的 tlias-web-management 项目打包为镜像,并部署。
-
步骤:
-
修改项目的配置文件,修改数据库服务地址(打包 package)。
- 编写 Dockerfile 文件(AI 辅助)。
- 构建 Docker 镜像,部署 Docker 容器,运行测试。
1). 修改项目的配置文件,修改数据库服务地址(打包 package)。
然后处理一下日志的生成位置.
然后,执行 maven 中的 package 生命周期,进行打包(跳过测试),并将打包后的 jar 包命名为 tlias.jar 。

2). 编写 Dockerfile 文件(AI 辅助)。
文件名 Dockerfile:
# 使用 CentOS 7 作为基础镜像
FROM centos:7
# 添加 JDK 到镜像中
COPY jdk17.tar.gz /usr/local/
RUN tar -xzf /usr/local/jdk17.tar.gz -C /usr/local/ && rm /usr/local/jdk17.tar.gz
# 设置环境变量
ENV JAVA_HOME=/usr/local/jdk-17.0.10
ENV PATH=$JAVA_HOME/bin:$PATH
# 阿里云OSS环境变量
ENV OSS_ACCESS_KEY_ID=LTAI5tP6dc4cvccdvvySE39X
ENV OSS_ACCESS_KEY_SECRET=ZSyIT31qhxIkS0dH1H9WzHqPiyM3Ot
#统一编码
ENV LANG=en_US.UTF-8
ENV LANGUAGE=en_US:en
ENV LC_ALL=en_US.UTF-8
# 创建应用目录
RUN mkdir -p /tlias
WORKDIR /tlias
# 复制应用 JAR 文件到容器
COPY tlias.jar tlias.jar
# 暴露端口
EXPOSE 8080
# 运行命令
ENTRYPOINT ["java","-jar","/tlias/tlias.jar"]
由于项目要运行,需要依赖 jdk 的环境,所以这里我们需要将 tlias.jar,jdk17.tar.gz,Dockerfile 三个文件,上传到 Linux 服务器的 /root/tlias 目录下(如果没有这个目录,提前创建好)。

3). 构建 Docker 镜像,部署 Docker 容器,运行测试。
- 构建 Docker 镜像
docker build -t tlias:1.0 .

- 部署 Docker 容器
docker run -d --name tlias-server --network itheima -p 8080:8080 tlias:1.0
[!TIP] --network itheima :将创建的容器,加入到 itheima 网络,就可以和 itheima 网络中的容器通信了。

通过 docker logs -f 容器名,就可以查看容器的运行日志。

这样后端服务,就已经启动起来了。
部署前端¶
- 需求:创建一个新的 nginx 容器,将资料中提供的前端项目的静态资源部署到 nginx 中。
-
步骤:
-
在宿主机上准备静态文件及配置文件存放目录(在
/usr/local目录下创建tlias-web目录)。-v /usr/local/tlias-web/html:/usr/share/nginx/html-v /usr/local/tlias-web/conf/nginx.conf:/etc/nginx/nginx.conf -
部署 nginx 容器 注意,nginx.conf里的反向代理的路径要改成tlias-server:8080
1). 部署 nginx 容器(设置目录映射)。 -
将资 资料/04. 项目部署/前端项目 中的 目录
html和 配置文件存放目录conf,上传至服务器端的/usr/local/tlias-web目录下。

- 执行如下命令,部署 nginx 容器
docker run -d \
--name nginx-tlias \
-v /usr/local/tlias-web/html:/usr/share/nginx/html \
-v /usr/local/tlias-web/conf/nginx.conf:/etc/nginx/nginx.conf \
--network itheima \
-p 80:80 \
nginx:1.20.2
前后端都部署完毕后,就可以打开浏览器,来测试一下。访问前端的 nginx 服务器 。
- ps:docker logs -f tlias-server 就能看到容器内部的日志.-f选项表示查看文件的末尾,就能看到server的springboot输出日志
DockerCompose¶
大家可以看到,我们部署一个简单的 java 项目,其中包含 3 个容器:
- MySQL
- Nginx
- Java 项目
而稍微复杂的项目,其中还会有各种各样的其它中间件,需要部署的东西远不止 3 个。而且有时候,容器的启动还有一定的依赖关系,这样的话就更麻烦了.如果还像之前那样手动的逐一部署,就太麻烦了。
而 Docker Compose 就可以帮助我们实现多个相互关联的 Docker 容器的快速部署。它允许用户通过一个单独的 docker-compose.yml 模板文件(YAML 格式)来定义一组相关联的应用容器。
基本语法¶
docker-compose 文件中可以定义多个相互关联的应用容器,每一个应用容器被称为一个服务(service)。由于 service 就是在定义某个应用的运行时参数,因此与 docker run 参数非常相似。
举例来说,用 docker run 部署 MySQL 的命令如下:
docker run -d \
--name nginx-tlias \
-p 80:80 \
-v /usr/local/app/html:/usr/share/nginx/html \
-v /usr/local/app/conf/nginx.conf:/etc/nginx/nginx.conf \
--network itheima \
nginx:1.20.2
如果用 docker-compose.yml 文件来定义,就是这样:
services:
mysql:
image: "nginx:1.20.2"
container_name: nginx-tlias
ports:
- "80:80"
volumes:
- "/usr/local/app/html:/usr/share/nginx/html"
- "/usr/local/app/conf/nginx.conf:/etc/nginx/nginx.conf"
networks:
- itheima
networks:
itheima:
name: itheima
| **docker run 参数** | **docker compose 指令** | **说明** |
| --name | container_name | 容器名称 |
| -p | ports | 端口映射 |
| -e | environment | 环境变量 |
| -v | volumes | 数据卷配置 |
| --network | networks | 网络 |
明白了其中的对应关系,相信编写 docker-compose 文件应该难不倒大家。
- 不过要提前准备好一些资源:tlias.sql,jdk17包,jar包,dockerfile,前端项目打包文件,nginx.conf,然后根据下面的compose把这些文件放到对应的位置去.
- 我们从下面也可以看到,dockercompose除了能帮我们一件运行容器,还能帮我们打包镜像(build)
services:
mysql:
image: mysql:8
container_name: mysql
ports:
- "3307:3306"
environment:
TZ: Asia/Shanghai
MYSQL_ROOT_PASSWORD: 123
volumes:
- "/usr/local/app/mysql/conf:/etc/mysql/conf.d"
- "/usr/local/app/mysql/data:/var/lib/mysql"
- "/usr/local/app/mysql/init:/docker-entrypoint-initdb.d"
networks:
- tlias-net//ps:这里的这个名称不是指网络的name,而是下面的networks选项里的那个tlias-net.
tlias:
build: //这样构建的镜像,通过dockerimages查看时,镜像的名字是创建时dockercompose文件所在目录名字-tlias
context: .//表示dockerfile就在当前目录
dockerfile: Dockerfile
container_name: tlias-server
ports:
- "8080:8080"
networks:
- tlias-net
depends_on:
- mysql//这里还能添加容器运行的顺序
nginx:
image: nginx:1.20.2
container_name: nginx
ports:
- "80:80"
volumes:
- "/usr/local/app/nginx/conf/nginx.conf:/etc/nginx/nginx.conf"
- "/usr/local/app/nginx/html:/usr/share/nginx/html"
depends_on:
- tlias
networks:
- tlias-net
networks:
tlias-net:
name: itheima
基础命令¶
编写好 docker-compose.yml 文件,就可以部署项目了。语法如下:
docker compose [OPTIONS] [COMMAND]
其中,OPTIONS 和 COMMAND 都是可选参数,比较常见的有:
| **类型** | **参数或指令** | **说明** |
| Options | -f | 指定compose文件的路径和名称 |
| -p | 指定project名称。project就是当前compose文件中设置的多个service的集合,是逻辑概念 | |
| Commands | up | 创建并启动所有service容器 |
| down | 停止并移除所有容器、网络 | |
| ps | 列出所有启动的容器 | |
| logs | 查看指定容器的日志 | |
| stop | 停止容器 | |
| start | 启动容器 | |
| restart | 重启容器 | |
| top | 查看运行的进程 | |
| exec | 在指定的运行中容器中执行命令 |
也可以(经常)加个-d后台运行
操作演示¶
1). 在 Linux 服务器的 /usr/local 目录下创建目录 app,并切换到 /usr/local/app 目录。
2). 上传资料中提供的 "资料/05. Docker Compose" 中的文件及文件夹到 /usr/local/app 目录中,如下所示:

注意,资料中提供的 Dockerfile 文件中的阿里云 OSS 的 AccessKeyId,AccessKeySecret 需要替换成自己的。
3). 执行如下指令,基于 DockerCompose 部署项目。
docker compose up -d

项目启动完毕之后,就可以启动服务器测试喽 。

附录:Docker 安装¶
卸载旧版¶
首先如果系统中已经存在旧的 Docker,则先卸载:
yum remove docker \
docker-client \
docker-client-latest \
docker-common \
docker-latest \
docker-latest-logrotate \
docker-logrotate \
docker-engine \
docker-selinux
配置 Docker 的 yum 库¶
首先要安装一个 yum 工具
sudo yum install -y yum-utils device-mapper-persistent-data lvm2
安装成功后,执行命令,配置 Docker 的 yum 源(已更新为阿里云源):
sudo yum-config-manager --add-repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
sudo sed -i 's+download.docker.com+mirrors.aliyun.com/docker-ce+' /etc/yum.repos.d/docker-ce.repo
更新 yum,建立缓存
sudo yum makecache fast
安装 Docker¶
最后,执行命令,安装 Docker
yum install -y docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin
启动和校验¶
# 启动Docker
systemctl start docker
# 停止Docker
systemctl stop docker
# 重启
systemctl restart docker
# 设置开机自启
systemctl enable docker
# 执行docker ps命令,如果不报错,说明安装启动成功
docker ps
配置镜像加速¶
镜像地址可能会变更,如果失效可以百度找最新的 docker 镜像。
配置镜像步骤如下:
# 创建目录
rm -f /etc/docker/daemon.json
# 复制内容
tee /etc/docker/daemon.json <<-'EOF'
{
"registry-mirrors": [
"http://hub-mirror.c.163.com",
"https://mirrors.tuna.tsinghua.edu.cn",
"http://mirrors.sohu.com",
"https://ustc-edu-cn.mirror.aliyuncs.com",
"https://ccr.ccs.tencentyun.com",
"https://docker.m.daocloud.io",
"https://docker.awsl9527.cn"
]
}
EOF
# 重新加载配置
systemctl daemon-reload
# 重启Docker
systemctl restart docker