Docker Compose:链接外部容器的几种方式

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介:

在Docker中,容器之间的链接是一种很常见的操作:它提供了访问其中的某个容器的网络服务而不需要将所需的端口暴露给Docker Host主机的功能。Docker Compose中对该特性的支持同样是很方便的。然而,如果需要链接的容器没有定义在同一个 docker-compose.yml 中的时候,这个时候就稍微麻烦复杂了点。

在不使用Docker Compose的时候,将两个容器链接起来使用 —link 参数,相对来说比较简单,以 nginx 镜像为例子:

 
 
  1. docker run --rm --name test1 -d nginx  #开启一个实例test1 
  2. docker run --rm --name test2 --link test1 -d nginx #开启一个实例test2并与test1建立链接 

这样, test2 与 test1 便建立了链接,就可以在 test2 中使用访问 test1 中的服务了。

如果使用Docker Compose,那么这个事情就更简单了,还是以上面的 nginx 镜像为例子,编辑 docker-compose.yml 文件为:

 
 
  1. version: "3" 
  2. services: 
  3.  test2: 
  4.  image: nginx 
  5.  depends_on: 
  6.  - test1 
  7.  links: 
  8.  - test1 
  9.  test1: 
  10.  image: nginx 

最终效果与使用普通的Docker命令 docker run xxxx 建立的链接并无区别。这只是一种最为理想的情况。

  1. 如果容器没有定义在同一个 docker-compose.yml 文件中,应该如何链接它们呢?
  2. 又如果定义在 docker-compose.yml 文件中的容器需要与 docker run xxx 启动的容器链接,需要如何处理?

针对这两种典型的情况,下面给出我个人测试可行的办法:

  1. 方式一:让需要链接的容器同属一个外部网络

我们还是使用nginx镜像来模拟这样的一个情景:假设我们需要将两个使用Docker Compose管理的nignx容器( test1 和 test2 )链接起来,使得 test2 能够访问 test1 中提供的服务,这里我们以能ping通为准。

首先,我们定义容器 test1 的 docker-compose.yml 文件内容为:

 
 
  1. version: "3" 
  2. services: 
  3.  test2: 
  4.  image: nginx 
  5.  container_name: test1 
  6.  networks: 
  7.  - default 
  8.  - app_net 
  9. networks: 
  10.  app_net: 
  11.  external: true 

容器 test2 内容与 test1 基本一样,只是多了一个 external_links ,需要特别说明的是: 最近发布的Docker版本已经不需要使用external_links来链接容器,容器的DNS服务可以正确的作出判断 ,因此如果你你需要兼容较老版本的Docker的话,那么容器 test2 的 docker-compose.yml文件内容为:

 
 
  1. version: "3" 
  2. services: 
  3.  test2: 
  4.  image: nginx 
  5.  networks: 
  6.  - default 
  7.  - app_net 
  8.  external_links: 
  9.  - test1 
  10.  container_name: test2 
  11. networks: 
  12.  app_net: 
  13.  external: true 

否则的话, test2 的 docker-compose.yml 和 test1 的定义完全一致,不需要额外多指定一个 external_links 。相关的问题请参见stackoverflow上的相关问题: docker-compose + external container

正如你看到的那样,这里两个容器的定义里都使用了同一个外部网络 app_net ,因此,我们需要在启动这两个容器之前通过以下命令再创建外部网络:

 
 
  1. docker network create app_net 

之后,通过 docker-compose up -d 命令启动这两个容器,然后执行 docker exec -it test2 ping test1 ,你将会看到如下的输出:

 
 
  1. docker exec -it test2 ping test1 
  2. PING test1 (172.18.0.2): 56 data bytes 
  3. 64 bytes from 172.18.0.2: icmp_seq=0 ttl=64 time=0.091 ms 
  4. 64 bytes from 172.18.0.2: icmp_seq=1 ttl=64 time=0.146 ms 
  5. 64 bytes from 172.18.0.2: icmp_seq=2 ttl=64 time=0.150 ms 
  6. 64 bytes from 172.18.0.2: icmp_seq=3 ttl=64 time=0.145 ms 
  7. 64 bytes from 172.18.0.2: icmp_seq=4 ttl=64 time=0.126 ms 
  8. 64 bytes from 172.18.0.2: icmp_seq=5 ttl=64 time=0.147 ms 

证明这两个容器是成功链接了,反过来在 test1 中ping test2 也是能够正常ping通的。

如果我们通过 docker run --rm --name test3 -d nginx 这种方式来先启动了一个容器( test3 )并且没有指定它所属的外部网络,而需要将其与 test1 或者 test2 链接的话,这个时候手动链接外部网络即可:

 
 
  1. docker network connect app_net test3 

这样,三个容器都可以相互访问了。

方式二:更改需要链接的容器的网络模式

通过更改你想要相互链接的容器的网络模式为 bridge ,并指定需要链接的外部容器( external_links )即可。与同属外部网络的容器可以相互访问的链接方式一不同,这种方式的访问是单向的。

还是以nginx容器镜像为例子,如果容器实例 nginx1 需要访问容器实例 nginx2 ,那么 nginx2 的 doker-compose.yml 定义为:

 
 
  1. version: "3" 
  2. services: 
  3.  nginx2: 
  4.  image: nginx 
  5.  container_name: nginx2 
  6.  network_mode: bridge 

与其对应的, nginx1 的 docker-compose.yml 定义为:

 
 
  1. version: "3" 
  2. services: 
  3.  nginx1: 
  4.  image: nginx 
  5.  external_links: 
  6.  - nginx2 
  7.  container_name: nginx1 
  8.  network_mode: bridge 

需要特别说明的是,这里的 external_links 是不能省略的,而且 nginx1 的启动必须要在 nginx2 之后,否则可能会报找不到容器 nginx2 的错误。

接着我们使用ping来测试下连通性:

 
 
  1. $ docker exec -it nginx1 ping nginx2  # nginx1 to nginx2 
  2. PING nginx2 (172.17.0.4): 56 data bytes 
  3. 64 bytes from 172.17.0.4: icmp_seq=0 ttl=64 time=0.141 ms 
  4. 64 bytes from 172.17.0.4: icmp_seq=1 ttl=64 time=0.139 ms 
  5. 64 bytes from 172.17.0.4: icmp_seq=2 ttl=64 time=0.145 ms 
  6.  
  7. $ docker exec -it nginx2 ping nginx1 #nginx2 to nginx1 
  8. ping: unknown host 

以上也能充分证明这种方式是属于单向联通的。

在实际应用中根据自己的需要灵活的选择这两种链接方式,如果想偷懒的话,大可选择第二种。不过我更推荐第一种,不难看出无论是联通性还是灵活性,较为更改网络模式的第二种都更为友好。 


本文作者:佚名

来源:51CTO

相关文章
|
17天前
|
Docker 容器
进入Docker容器中
进入Docker容器中
32 2
|
22天前
|
存储 机器学习/深度学习 中间件
快速上手 Elasticsearch:Docker Compose 部署详解
本文介绍了如何使用Docker Compose快速搭建Elasticsearch学习环境。Elasticsearch是一款用于实时搜索和分析的分布式中间件,适用于多种场景,如搜索、日志分析、机器学习等。首先,创建docker网络,拉取最新版8.12.2镜像。接着,编写docker-compose.yml文件,配置单节点集群,设置端口映射、内存限制及数据卷挂载。然后,创建并配置数据卷目录,允许远程访问和跨域。最后,启动服务并验证,通过浏览器访问确认服务运行正常。本文为初学者提供了一个简便的Elasticsearch部署方法。
125 4
快速上手 Elasticsearch:Docker Compose 部署详解
|
27天前
|
关系型数据库 MySQL Java
Docker Compose详细教程(从入门到放弃)
Docker Compose详细教程(从入门到放弃)
112 0
|
6天前
|
Linux Docker 容器
docker 容器常用命令
docker 容器常用命令
11 0
|
14天前
|
Kubernetes 网络协议 Docker
Docker 容器的DNS
Docker 容器的DNS
23 1
|
17天前
|
关系型数据库 MySQL Nacos
【深入浅出Nacos原理及调优】「实战开发专题」采用Docker容器进行部署和搭建Nacos服务以及“坑点”
【深入浅出Nacos原理及调优】「实战开发专题」采用Docker容器进行部署和搭建Nacos服务以及“坑点”
42 1
|
22天前
|
监控 数据可视化 虚拟化
Docker容器常用命令笔记分享
Docker容器常用命令笔记分享
49 2
|
1月前
|
Java Go 开发者
Docker容器技术简介及其与Go语言的结合点
【2月更文挑战第23天】本文首先概述了Docker容器技术的核心概念和优势,接着探讨了Go语言与Docker容器技术的结合点。通过阐述Docker的轻量级、可移植性和版本控制等特性,以及Go语言在容器化应用中的优势,本文旨在说明两者结合能够实现更高效、灵活的应用开发和部署。
|
1月前
|
Oracle 关系型数据库 数据库