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

在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

时间: 2024-11-03 22:00:26

Docker Compose:链接外部容器的几种方式的相关文章

替代Docker Compose实现容器双向联通的三种方法

Docker 是目前最热门的技术平台之一,他在产生后很短的时间内就获得了社会的广泛关注.简单的说,他使得开发者和系统管理员能够用一种简易的方法去部署分布式应用.Docker 的生态系统非常庞大,有很多的工具协同工作,比如最常用的工具之一:Docker Compose.他使你可以在单个文件中定义并运行多容器应用,然后通过一个命令执行. 一个 docker-compose.yml 文件看起来是这样的: links 选项使容器能够在一个运行中创建的内部网络中通讯,并且在运行终止后销毁.在上面的例子中,

Spring Boot与Docker(四):额外的微服务、更新容器、Docker Compose和负载均衡

本文讲的是Spring Boot与Docker(四):额外的微服务.更新容器.Docker Compose和负载均衡,[编者的话]本篇是<使用Spring Boot和Docker构建微服务架构>系列的第四篇,本篇我们我们将添加一些额外的服务/容器,并且更新容器,采用Docker Compose以及使用HAProxy容器进行负载均衡.原文作者为3Pillar环球旗下美国Adbanced技术集团的总监Dan Greene,Dan有十八年的软件设计和开发经验,包括在电子商务.B2B集成.空间分析.S

开发Docker容器的8种模式

[编者按]Vidar Hokstad 在Docker使用方面非常有经验,尤其在没有数据丢失前提下,使用Docker创建可重复build上经验丰富,在本博客中,他总结了开发Docker容器的8种模式. 以下为译文: Docker现在成了我最喜欢的工具,在本文中,我将概述一些在我使用Docker过程中反复出现的模式.我不期待它们能给你带来多少惊喜,但我希望这些能对你有用,我非常愿意与你交流在使用Docker过程中碰到的模式. 我所有Docker实验的基础是保持volume状态不变,以便Docker容

容器间通信的三种方式 - 每天5分钟玩转 Docker 容器技术(35)

容器之间可通过 IP,Docker DNS Server 或 joined 容器三种方式通信. IP 通信 从上一节的例子可以得出这样一个结论:两个容器要能通信,必须要有属于同一个网络的网卡. 满足这个条件后,容器就可以通过 IP 交互了.具体做法是在容器创建时通过 --network 指定相应的网络,或者通过 docker network connect 将现有容器加入到指定网络.可参考上一节 httpd 和 busybox 的例子,这里不再赘述. Docker DNS Server 通过 I

两种方式创建docker镜像的启动容器时区别介绍(总结篇)_docker

Docker是建立在Linux内核基础上的,在目前的主流Linux系统中,都已经原生支持了Docker且使用体验也最好,当然,在Windows平台和MacOS系统中也支持Docker,只是需要使用类似Boot2Docker等虚拟化工具来提供Linux支持. 下面重点给大家介绍基于两种创建docker镜像的启动容器时区别,感兴趣的朋友可以跟着小编一起学习! 1.凡是用docker commit生成的镜像启动的时候可以加载一个启动自己应用的脚本,例如: docker run -d -P tomcat

Docker容器进入的4种方式(转)

这个文章不错,几种方式及使用范围都讲得清楚. 另外,还有一个文章说了为什么docker中并不合适ssh服务安装的情况. 我现在也是用第四种方式, docker exec -it [docker容器ID] /bin/bash Docker容器进入的4种方式 http://www.cnblogs.com/xhyan/p/6593075.html 为什么不需要在 Docker 容器中运行 sshd http://www.oschina.net/translate/why-you-dont-need-t

利用Docker和阿里云容器服务部署高可用Ghost博客集群

简介 Ghost是一个流行的开源博客平台(Open source blogging platform),基于 Node.js 构建,博客内容默认采用 Markdown 语法书写,给用户提供一种更加纯粹的内容写作与发布平台. Ghost的部署和运维需要一定的Web开发基础,利用Docker技术可以大大简化Ghost的部署和更新.Docker Hub上面也提供了Ghost官方镜像 使用Docker镜像,不懂得Node.Js的同学也可以分分钟在本地或阿里云容器服务上搭建起一个单节点的Ghost博客,但

通过Docker Compose及NGINX反向代理实现ASP.NET5应用的负载均衡

本文以ASP.NET应用为例,讲述了如何通过NGINX访问运行在本机上的ASP.NET 5服务,并采用Docker Compose对ASP服务进行编排,同时提供简单的负载均衡机制. ASP.NET 5可以在很多操作系统下运行,也支持IIS等多种不同的web服务器.网上关于ASP.NET 5的教程有很多,这里我就不多说了,大家可以自己去搜索.今天我们要讲的是如何通过NGINX访问运行在本机上的ASP.NET 5服务,并采用Docker Compose对ASP服务进行编排,同时提供简单的负载均衡机制

使用Docker Compose部署基于Sentinel的高可用Redis集群

大家一定非常熟悉如何利用Docker启动单个Redis容器用于开发环境,本文将介绍如何利用Docker Compose模板在本机和云端部署基于Sentinel的高可用Redis 3集群. Redis集群可以在一组redis节点之间实现高可用性和sharding.今天我们重点围绕master-slave的高可用模式来进行讨论,在集群中会有1个master和多个slave节点.当master节点失效时,应选举出一个slave节点作为新的master.然而Redis本身(包括它的很多客户端)没有实现自