詳解docker compose 用法
docker compose的使用場景
我們開發(fā)的時(shí)候,一個(gè)應(yīng)用往往依賴多個(gè)服務(wù)。采用傳統(tǒng)的docker run方式,要挨個(gè)啟動多個(gè)服務(wù),甚至需要配置對應(yīng)的網(wǎng)絡(luò),過程比較繁瑣,很不方便。 docker compose旨在通過將多服務(wù)的構(gòu)建和依賴關(guān)系都編寫在docker-compose.yml中,通過docker-compose命令,即可完成對整個(gè)服務(wù)集群的啟動,關(guān)閉等操作。
一個(gè)基本的demo演示
demo的功能是一個(gè)簡單的python程序,暴露一個(gè)web服務(wù)。該服務(wù)用于統(tǒng)計(jì)當(dāng)前服務(wù)被訪問的次數(shù)。次數(shù)的累加和存儲,都是基于redis進(jìn)行的。也即該程序本身除了自己的服務(wù),還要依賴一個(gè)redis服務(wù)。以下是詳細(xì)步驟
找一個(gè)目錄,在其中創(chuàng)建一個(gè)python文件app.py
import time
import redis
from flask import Flask
app = Flask(__name__)
cache = redis.Redis(host='redis', port=6379)
def get_hit_count():
retries = 5
while True:
try:
return cache.incr('hits')
except redis.exceptions.ConnectionError as exc:
if retries == 0:
raise exc
retries -= 1
time.sleep(0.5)
@app.route('/')
def hello():
count = get_hit_count()
return 'Hello World! I have been seen {} times.\n'.format(count)
在相同的文件夾下,創(chuàng)建requirements.txt文件
requirements.txt文件用來聲明python程序需要使用到的依賴lib,有點(diǎn)像java中的maven pom文件。上述代碼使用的組件有flask和redis。所以requirements.txt文件內(nèi)容為
flask redis
在相同的文件夾下,創(chuàng)建Dockerfile
Dockerfile用來將我們的程序構(gòu)建成一個(gè)docker 鏡像,即docker image。一般Dockerfile中會定義我們的代碼運(yùn)行的基本環(huán)境,程序啟動命令,執(zhí)行端口等。本例的Dockerfile如下
FROM python:3.7-alpine WORKDIR /code ENV FLASK_APP=app.py ENV FLASK_RUN_HOST=0.0.0.0 COPY requirements.txt requirements.txt RUN pip install -r requirements.txt EXPOSE 5000 COPY . . CMD ["flask", "run"]
在相同的文件下,創(chuàng)建docker-compose.yml文件
上述幾步完成后,我們得到了我們服務(wù)本身的docker化執(zhí)行的能力。但該服務(wù)依賴redis service。所以我們通過docker-compose.yml來組織服務(wù)的依賴關(guān)系,內(nèi)容如下:
version: "3.8" services: web: build: . ports: - "5000:5000" redis: image: "redis:alpine"
文件中定義了兩個(gè)服務(wù)web和redis , web中的build:. 會在當(dāng)前目錄下基于前面定義的Dockerfile將我們的代碼構(gòu)建成一個(gè)image,然后啟動成一個(gè)container時(shí),會對外暴露5000端口,映射到當(dāng)前宿主機(jī)的端口也是5000
redis服務(wù)直接使用現(xiàn)成的image redis:alpine,沒有指定端口,將暴露redis的默認(rèn)端口
基礎(chǔ)運(yùn)維
所有docker-compose相關(guān)的命令,都要在docker-compose.yml所在的路徑下執(zhí)行才行
啟動基于docker-compose.yml編織好的服務(wù)
在docker-compose.yml所在的目錄,使用命令docker-compose up即可。但該命令在console關(guān)閉時(shí),對應(yīng)的docker service也會被關(guān)閉。可以是使用docker-compose up -d 以后臺detach模式去執(zhí)行。
docker-compose up 也可以單獨(dú)啟動compolse file中的某個(gè)服務(wù)及其依賴
查看compose服務(wù)對應(yīng)的容器服務(wù)列表
docker-compose ps
輸出結(jié)果樣例
# docker-compose ps
Name Command State Ports
----------------------------------------------------------------------------------------------
docker_compose_learn_redis_1 docker-entrypoint.sh redis ... Up 6379/tcp
docker_compose_learn_web_1 flask run Up 0.0.0.0:5000->5000/tcp
服務(wù)前綴docker_compose_learn是當(dāng)前項(xiàng)目的名稱。項(xiàng)目名稱可以通過環(huán)境變量COMPOSE_PROJECT_NAME來指定,如果未指定,默認(rèn)的項(xiàng)目名稱為compose 文件所在文件夾的名字。本例中的文件夾名為docker_compose_learn
當(dāng)然compose的一系列服務(wù),最終也是啟動了一系列的container. 所以也可使用docker container命令族進(jìn)行管理,但是太麻煩
停止service的container
需要在docker-compose.yml所在的路徑,使用命令docker-compose stop
停止service的container,并且刪除對應(yīng)的container
需要在docker-compose.yml所在的路徑,使用命令
docker-compose down
停止service的container,并且刪除對應(yīng)的container和對應(yīng)的volumes數(shù)據(jù)
需要在docker-compose.yml所在的路徑,使用命令
docker-compose down --volumes
該命令并不會刪除掛載的宿主操作系統(tǒng)的文件。
如何登進(jìn)對應(yīng)的service
想要登錄到compose中,具體某個(gè)service的命令行,使用如下命令
docker-compose exec ***servicename*** bash

如何顯示指定compose file
docker-compose -f docker-compose.yml -f docker-compose.admin.yml run backup_db
docker-compose up和docker-compose run的區(qū)別
docker-compose up會基于compose file 啟動所有的的服務(wù),并對外暴露端口
docker-compose run需要指定特定的服務(wù)進(jìn)行啟動,比如docker-compose run web bash只會啟動compolse文件中的web服務(wù)和其依賴的service,并且不會對外暴露端口,以免跟docker-compose up啟動的服務(wù)端口沖突。
docker-compose run僅用在臨時(shí)啟動某個(gè)服務(wù)定位問題的場景
一些擴(kuò)展知識點(diǎn)
環(huán)境變量
docker-compose.yml的內(nèi)容本身可以使用變量占位符,其具體的變量值定義在具體的環(huán)境變量中,這樣方便同一份docker-compose.yml文件在不同的環(huán)境有不同的執(zhí)行行為。典型的,我們希望依賴服務(wù)的image的tag版本,隨環(huán)境不同而不同。
那么我們在docker-compose.yml對應(yīng)的服務(wù)配置中以占位符配置其tag,以下用${TAG}配置web服務(wù)的image tag
web:
image: "webapp:${TAG}"
除了指定以的變量意外,還有多個(gè)docker內(nèi)置的變量可以設(shè)置,他們用來配置docker的或者docker compose的執(zhí)行行為。這些內(nèi)置變量是
- COMPOSE_API_VERSION
- COMPOSE_CONVERT_WINDOWS_PATHS
- COMPOSE_FILE
- COMPOSE_HTTP_TIMEOUT
- COMPOSE_TLS_VERSION
- COMPOSE_PROJECT_NAME
- DOCKER_CERT_PATH
- DOCKER_HOST
- DOCKER_TLS_VERIFY
具體含義參見;https://docs.docker.com/compose/reference/envvars/
以占位符TAG為例,講解變量的設(shè)置可以有以下幾種方式
在docker-compose.yml中執(zhí)行
在compolse文件中,通過environment配置項(xiàng)指定
web:
image: "webapp:${TAG}"
environment:
- TAG=dev
在執(zhí)行docker-compose 命令之前設(shè)置shell環(huán)境變量
$ export TAG=v2.0 $ docker-compose up
通過env_file文件設(shè)置
docker-compose up默認(rèn)會找命令執(zhí)行路徑下的.env文件,去其中找變量替換的值,.env文件以key=value的形式配置。例如
TAG=dev
如果環(huán)境變量的名字不為.env或不在當(dāng)前命令執(zhí)行的路徑下,可以在使用--env-file參數(shù)顯示加載
docker-compose --env-file ./config/.env.dev up
直接在compose 文件中,指定其加載的env_file
version: '3' services: api: image: 'node:6-alpine' env_file: - ./Docker/api/api.env environment: - NODE_ENV=production
以上變量值設(shè)置優(yōu)先級從高到底
查看最終生效的環(huán)境變量
如果不確定最終生效環(huán)境變量是什么樣,可以使用以下命令來查看
docker-compose run web env
項(xiàng)目名設(shè)定
一個(gè)compose對應(yīng)的一組服務(wù)有一個(gè)公用的項(xiàng)目名(project name), 它會體現(xiàn)在compose服務(wù)的容器名前綴中,網(wǎng)絡(luò)前綴中。
項(xiàng)目名稱可以通過環(huán)境變量COMPOSE_PROJECT_NAME來指定,如果未指定,默認(rèn)的項(xiàng)目名稱為compose 文件所在文件夾的名字。
網(wǎng)絡(luò)
默認(rèn)網(wǎng)絡(luò)
默認(rèn)情況下,compose中的多個(gè)服務(wù)會加入一個(gè)名為default的網(wǎng)絡(luò)。這些服務(wù)在default網(wǎng)絡(luò)中是互通的。該default網(wǎng)絡(luò)的全稱是以compose文件所在文件夾名字做為前綴。比如文件夾為hello_world的compose。其一組服務(wù)對應(yīng)的網(wǎng)絡(luò)名為:hello_world_default。 這組service在該網(wǎng)絡(luò)中,以compose文件中的第二組端口通信。
version: "3" services: web: build: . ports: - "8000:8000" db: image: postgres ports: - "8001:5432"
比如上述配置中,在hello_world_default網(wǎng)絡(luò)中,web服務(wù)使用8000端口和db服務(wù)的5432端口通信。第一組端口8000和8001是宿主機(jī)訪問web和db服務(wù)的端口。
對默認(rèn)網(wǎng)絡(luò)進(jìn)行獨(dú)立配置
如果想改變默認(rèn)網(wǎng)絡(luò)的配置,可以在compose文件中,單獨(dú)通過networks項(xiàng)來改變,比如以下改變默認(rèn)網(wǎng)絡(luò)驅(qū)動
networks: default: # Use a custom driver driver: custom-driver-1
配置和使用非默認(rèn)網(wǎng)絡(luò)
定義多個(gè)網(wǎng)絡(luò),并使用
version: "3" services: proxy: build: ./proxy networks: - frontend app: build: ./app networks: - frontend - backend db: image: postgres networks: - backend networks: frontend: # Use a custom driver driver: custom-driver-1 backend: # Use a custom driver which takes special options driver: custom-driver-2 driver_opts: foo: "1" bar: "2"
上述配置定義了兩個(gè)網(wǎng)絡(luò),frontend 和 backend。其中app 能訪問這兩個(gè)網(wǎng)絡(luò),proxy服務(wù)只能訪問frontend網(wǎng)絡(luò),db只能訪問backend網(wǎng)絡(luò)
多service的執(zhí)行順序
一個(gè)compose的多個(gè)service可能會有依賴關(guān)系,比如web服務(wù)依賴db服務(wù),我們希望先啟動db服務(wù),再啟動web服務(wù)。這種啟動的先后順序,也可以在compose文件中使用depends_on指定
version: "2" services: web: build: . ports: - "80:8000" depends_on: - "db" command: ["./wait-for-it.sh", "db:5432", "--", "python", "app.py"] db: image: postgres
docker compose的安裝
docker mac版和windows版,默認(rèn)都帶有docker compose 。 只有l(wèi)inux版需要單獨(dú)安裝
docker compose和docker stack的異同
- docker compose主要目標(biāo)是在同一臺機(jī)器上啟動并管理多個(gè)服務(wù)
- docker stack主要用于在多個(gè)機(jī)器上,啟動并管理多個(gè)服務(wù)
- docker compose 和docker stack都可以使用docker-compose.yml文件。雙方會自動忽略對自己不生效的配置
- docker compose的服務(wù)可以使用build動態(tài)構(gòu)建,而docker stack的服務(wù)只能基于image
參考資料
https://docs.docker.com/compose/gettingstarted/
https://docs.docker.com/compose/
https://stackoverflow.com/questions/43099408/whats-the-difference-between-a-stack-file-and-a-compose-file
https://nickjanetakis.com/blog/docker-tip-23-docker-compose-vs-docker-stack
https://vsupalov.com/difference-docker-compose-and-docker-stack/
https://stackoverflow.com/questions/33066528/should-i-use-docker-compose-up-or-run
到此這篇關(guān)于docker compose 用法的文章就介紹到這了,更多相關(guān)docker compose 用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解使用Dockerfile創(chuàng)建帶Apache服務(wù)的CentOS Docker鏡像
本篇文章主要介紹了使用Dockerfile創(chuàng)建帶Apache服務(wù)的CentOS Docker鏡像,具有一定的參考價(jià)值,有興趣的可以了解一下。2016-12-12
基于docker部署skywalking實(shí)現(xiàn)全鏈路監(jiān)控功能
這篇文章主要介紹了基于docker部署skywalking實(shí)現(xiàn)全鏈路監(jiān)控,skywalking提供了在很多不同的場景下用于觀察和監(jiān)控分布式系統(tǒng)的方式,文中給大家介紹如何快速部署skywalking全鏈路監(jiān)控,感興趣的朋友一起看看吧2022-03-03
Docker之開啟遠(yuǎn)程訪問的實(shí)現(xiàn)
這篇文章主要介紹了Docker之開啟遠(yuǎn)程訪問的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-05-05
在docker中部署tomcat并且部署java應(yīng)用程序的步驟詳解
本文給大家介紹如何在docker中部署tomcat及如何部署我們的應(yīng)用程序到docker容器的tomcat里,本文介紹的非常詳細(xì),具有參考借鑒價(jià)值,一起看看吧2016-10-10
docker容器內(nèi)網(wǎng)絡(luò)請求緩慢問題解決
在使用docker的過程中發(fā)現(xiàn)了幾個(gè)問題,在docker里進(jìn)行的網(wǎng)絡(luò)請求經(jīng)常會失敗,這篇文章主要介紹了docker容器內(nèi)網(wǎng)絡(luò)請求緩慢問題解決2019-01-01
Docker?部署?Nexus?Maven私服的詳細(xì)過程
Nexus?是一個(gè)強(qiáng)大的倉庫管理器,廣泛用于管理和組織軟件構(gòu)建過程中的依賴項(xiàng)和構(gòu)件,通過?Docker?部署?Nexus?私服,可以簡化安裝和管理過程,并提供更高的靈活性和可擴(kuò)展性,這篇文章主要介紹了Docker?部署?Nexus?Maven私服的詳細(xì)過程,需要的朋友可以參考下2024-08-08

