30 分鐘快速入門 Docker 教程
原文地址:梁桂釗的博客
博客地址:http://blog.720ui.com
歡迎關註公眾號:「服務端思維」。一群同頻者,一起成長,一起精進,打破認知的局限性。
一、歡迎來到 Docker 世界
1. Docker 與虛擬化
在沒有 Docker 的時代,我們會使用硬件虛擬化(虛擬機)以提供隔離。這裏,虛擬機通過在操作系統上建立了一個中間虛擬軟件層 Hypervisor ,並利用物理機器的資源虛擬出多個虛擬硬件環境來共享宿主機的資源,其中的應用運行在虛擬機內核上。但是,虛擬機對硬件的利用率存在瓶頸,因為虛擬機很難根據當前業務量動態調整其占用的硬件資源,因此容器化技術得以流行。其中,Docker 是一個開源的應用容器引擎,讓開發者可以打包他們的應用以及依賴包到一個可移植的容器中,然後發布到任何流行的 Linux 機器上。
Docker 容器不使用硬件虛擬化,它的守護進程是宿主機上的一個進程,換句話說,應用直接運行在宿主機內核上。因為容器中運行的程序和計算機的操作系統之間沒有額外的中間層,沒有資源被冗余軟件的運行或虛擬硬件的模擬而浪費掉。
Docker 的優勢不僅如此,我們來比較一番。
特性 | Docker | 虛擬機 |
---|---|---|
啟動速度 | 秒級 | 分鐘級 |
交付/部署 | 開發、測試、生產環境一致 | 無成熟體系 |
性能 | 近似物理機 | 性能損耗大 |
體量 | 極小(MB) | 較大(GB) |
遷移/擴展 | 跨平臺,可復制 | 較為復雜 |
2. 鏡像、容器和倉庫
Docker 由鏡像(Image)、容器(Container)、倉庫(Repository) 三部分組成。
Docker 的鏡像可以簡單的類比為電腦裝系統用的系統盤,包括操作系統,以及必要的軟件。例如,一個鏡像可以包含一個完整的 centos 操作系統環境,並安裝了 Nginx 和 Tomcat 服務器。註意的是,鏡像是只讀的。這一點也很好理解,就像我們刻錄的系統盤其實也是可讀的。我們可以使用 docker images
?來查看本地鏡像列表。
Docker 的容器可以簡單理解為提供了系統硬件環境,它是真正跑項目程序、消耗機器資源、提供服務的東西。例如,我們可以暫時把容器看作一個 Linux 的電腦,它可以直接運行。那麽,容器是基於鏡像啟動的,並且每個容器都是相互隔離的。註意的是,容器在啟動的時候基於鏡像創建一層可寫層作為最上層。我們可以使用 docker ps -a
Docker 的倉庫用於存放鏡像。這一點,和 Git 非常類似。我們可以從中心倉庫下載鏡像,也可以從自建倉庫下載。同時,我們可以把制作好的鏡像 commit 到本地,然後 push 到遠程倉庫。倉庫分為公開倉庫和私有倉庫,最大的公開倉庫是官方倉庫 Dock Hub,國內的公開倉庫也有很多選擇,例如阿裏雲等。
3. Docker 促使開發流程變更
筆者認為,Docker 對開發流程的影響在於使環境標準化。例如,原來我們存在三個環境:開發(日常)環境、測試環境、生產環境。這裏,我們對於每個環境都需要部署相同的軟件、腳本和運行程序,如圖所示。事實上,對於啟動腳本內容都是一致的,但是沒有統一維護,經常會出問題。此外,對於運行程序而言,如果所依賴的底層運行環境不一致,也會造成困擾和異常。
現在,我們通過引入 Docker 之後,我們只需要維護一個 Docker?鏡像。換句話說,多套環境,一個鏡像,實現系統級別的一次構建到處運行。此時,我們把運行腳本標準化了,把底層軟件鏡像化了,然後對於相同的將要部署的程序實行標準化部署。因此,Docker 為我們提供了一個標準化的運維模式,並固化運維步驟和流程。
通過這個流程的改進,我們更容易實現 DevOps 的目標,因為我們的鏡像生成後可以跑在任何系統,並快速部署。此外,使用 Docker 的很大動力是基於 Docker 實現彈性調度,以更充分地利用機器資源,節省成本。
哈哈,筆者在使用 Docker 過程中,還發現了一些很棒的收益點,例如我們發布回滾的時候只需要切換 TAG 並重啟即可。還比如,我們對環境升級,也只需要升級基礎鏡像,那麽新構建的應用鏡像,自動會引用新的版本。(歡迎補充~~~)
二、從搭建 Web 服務器開始說起
1. 環境先行,安裝 Docker
現在,我們需要安裝以下步驟安裝 Docker。
- 註冊帳號:在 https://hub.docker.com/ 註冊賬號。
- 下載安裝
官方下載地址:(Mac):https://download.docker.com/mac/stable/Docker.dmg
阿裏雲下載地址(Mac):http://mirrors.aliyun.com/docker-toolbox/mac/docker-for-mac/
阿裏雲下載地址(Windows): http://mirrors.aliyun.com/docker-toolbox/windows/docker-for-windows/ - 安裝指南
這裏,雙擊剛剛下載的 Doker.dmg 安裝包進行安裝。
安裝完成後啟動, Mac 頂部導航欄出現了一個圖標,通過菜單可以進行 docker 配置和退出等操作。
官方指南:https://docs.docker.com/install/
阿裏雲指南(Linux):https://yq.aliyun.com/articles/110806?spm=5176.8351553.0.0.468b1991jdT95t
- 設置加速服務
市面上有很多加速服務的提供商,如:DaoCloud,阿裏雲等。這裏,筆者使用的是阿裏雲。(註意的是,筆者操作系統是?Mac,其他操作系列參見阿裏雲操作文檔)?
右鍵點擊桌面頂欄的 docker 圖標,選擇 Preferences ,在 Daemon 標簽(Docker 17.03 之前版本為 Advanced 標簽)下的 Registry mirrors 列表中將<br />https://xxx.mirror.aliyuncs.com
?加到"registry-mirrors"的數組裏,點擊 Apply & Restart 按鈕,等待 Docker 重啟並應用配置的鏡像加速器。
阿裏雲操作文檔:https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors
- 查看版本
至此,我們已經安裝完成了。這裏,我們來查看版本。
docker version
查看結果,如下所示。
2. 實幹派,從搭建 Web 服務器開始
我們作為實幹派,那麽先來搭建一個 Web 服務器吧。然後,筆者帶你慢慢理解這個過程中,做了什麽事情。首先,我們需要拉取 centos 鏡像。
docker run -p 80 --name web -i -t centos /bin/bash
緊接著,我們安裝 nginx 服務器,執行以下命令:
rpm -ivh http://nginx.org/packages/centos/7/noarch/RPMS/nginx-release-centos-7-0.el7.ngx.noarch.rpm
安裝完 Nginx 源後,就可以正式安裝 Nginx 了。
yum install -y nginx
至此,我們再輸入 whereis nginx
?命令就可以看到安裝的路徑了。最後,我們還需要將 Nginx 跑起來。
nginx
現在,我們執行?ctrl + P + ?Q
?切換到後臺。然後,通過?docker ps -a
?來查看隨機分配的端口。
這裏,筆者分配的端口是 32769
?,那麽通過瀏覽器訪問 http://127.0.0.1:32769
?即可。
大功告成,哈哈哈~
3. 復盤理解全過程
現在,我們來理解下這個流程。首先,我們輸入 docker run -p 80 --name web -i -t centos /bin/bash
?命令會運行交互式容器,其中 -i
?選項告訴 Docker 容器保持標準輸入流對容器開放,即使容器沒有終端連接,另一個 -t
?選項告訴 Docker 為容器分配一個虛擬終端,以便於我們接下來安裝 Nginx 服務器。(筆者備註:Docker 還支持輸入 -d
?選項告訴 Docker 在後臺運行容器的守護進程)
Docker 會為我們創建的每一個容器自動生成一個隨機的名稱。事實上,這種方式雖然便捷,但是可讀性很差,並且對我們後期維護的理解成本會比較大。因此,我們通過 --name web
?選項告訴 Docker 創建一個名稱是 web
?的容器。此外,我們通過 -p 80
?告訴 Docker 開放 80 端口,那麽, Nginx 才可以對外通過訪問和服務。但是,我們的宿主機器會自動做端口映射,比如上面分配的端口是?32769
?,註意的是,如果關閉或者重啟,這個端口就變了,那麽怎麽解決固定端口的問題,筆者會在後面詳細剖析和帶你實戰。
這裏,還有一個非常重要的知識點 docker run
?。Docker 通過 run 命令來啟動一個新容器。Docker 首先在本機中尋找該鏡像,如果沒有安裝,Docker 在 Docker Hub 上查找該鏡像並下載安裝到本機,最後 Docker 創建一個新的容器並啟動該程序。
但是,當第二次執行??docker run
?時,因為?Docker 在本機中已經安裝該鏡像,所以 Docker 會直接創建一個新的容器並啟動該程序。
註意的是,docker run
?每次使用都會創建一個新的容器,因此,我們以後再次啟動這個容器時,只需要使用命令 docker start
? 即可。這裏, docker start
?的作用在用重新啟動已存在的鏡像,而docker run
?包含將鏡像放入容器中 docker create
?,然後將容器啟動?docker start
?,如圖所示。
現在,我們可以在上面的案例的基礎上,通過 exit
?命令關閉 Docker 容器。當然,如果我們運行的是後臺的守護進程,我們也可以通過 docker stop web
?來停止。註意的是,docker stop
?和?docker kill
?略有不同,docker stop
?發送 SIGTERM 信號,而?docker kill
?發送SIGKILL 信號。然後,我們使用?docker start
?重啟它。
docker start web
Docker 容器重啟後會沿用?docker run
?命令指定的參數來運行,但是,此時它還是後臺運行的。我們必須通過 docker attach
?命令切換到運行交互式容器。
docker attach web
4. 不止如此,還有更多命令
Docker 提供了非常豐富的命令。所謂一圖勝千言,我們可以從下面的圖片了解到很多信息和它們之前的用途。(可以直接跳過閱讀,建議收藏,便於擴展閱讀)
如果希望獲取更多信息,可以閱讀官方使用文檔。
Command | Description |
---|---|
docker attach | Attach local standard input, output, and error streams to a running container |
docker build | Build an image from a Dockerfile |
docker builder | Manage builds |
docker checkpoint | Manage checkpoints |
docker commit | Create a new image from a container’s changes |
docker config | Manage Docker configs |
docker container | Manage containers |
docker cp | Copy files/folders between a container and the local filesystem |
docker create | Create a new container |
docker deploy | Deploy a new stack or update an existing stack |
docker diff | Inspect changes to files or directories on a container’s filesystem |
docker engine | Manage the docker engine |
docker events | Get real time events from the server |
docker exec | Run a command in a running container |
docker export | Export a container’s filesystem as a tar archive |
docker history | Show the history of an image |
docker image | Manage images |
docker images | List images |
docker import | Import the contents from a tarball to create a filesystem image |
docker info | Display system-wide information |
docker inspect | Return low-level information on Docker objects |
docker kill | Kill one or more running containers |
docker load | Load an image from a tar archive or STDIN |
docker login | Log in to a Docker registry |
docker logout | Log out from a Docker registry |
docker logs | Fetch the logs of a container |
docker manifest | Manage Docker image manifests and manifest lists |
docker network | Manage networks |
docker node | Manage Swarm nodes |
docker pause | Pause all processes within one or more containers |
docker plugin | Manage plugins |
docker port | List port mappings or a specific mapping for the container |
docker ps | List containers |
docker pull | Pull an image or a repository from a registry |
docker push | Push an image or a repository to a registry |
docker rename | Rename a container |
docker restart | Restart one or more containers |
docker rm | Remove one or more containers |
docker rmi | Remove one or more images |
docker run | Run a command in a new container |
docker save | Save one or more images to a tar archive (streamed to STDOUT by default) |
docker search | Search the Docker Hub for images |
docker secret | Manage Docker secrets |
docker service | Manage services |
docker stack | Manage Docker stacks |
docker start | Start one or more stopped containers |
docker stats | Display a live stream of container(s) resource usage statistics |
docker stop | Stop one or more running containers |
docker swarm | Manage Swarm |
docker system | Manage Docker |
docker tag | Create a tag TARGET_IMAGE that refers to SOURCE_IMAGE |
docker top | Display the running processes of a container |
docker trust | Manage trust on Docker images |
docker unpause | Unpause all processes within one or more containers |
docker update | Update configuration of one or more containers |
docker version | Show the Docker version information |
docker volume | Manage volumes |
docker wait | Block until one or more containers stop, then print their exit codes |
官方閱讀鏈接:https://docs.docker.com/engine/reference/commandline/docker/
5. 進階:倉庫與軟件安裝的簡化
還記得筆者在文章開頭介紹的「鏡像、容器和倉庫」嗎?Docker 的倉庫用於存放鏡像。我們可以從中心倉庫下載鏡像,也可以從自建倉庫下載。同時,我們可以把制作好的鏡像從本地推送到遠程倉庫。
首先,筆者先引入一個知識點:Docker 的鏡像就是它的文件系統,一個鏡像可以放在另外一個鏡像的上層,那麽位於下層的就是它的父鏡像。所以,Docker 會存在很多鏡像層,每個鏡像層都是只讀的,並且不會改變。當我們創建一個新的容器時,Docker 會構建出一個鏡像棧,並在棧的最頂層添加一個讀寫層,如圖所示。
現在,我們可以通過 docker images
?命令查看本地的鏡像。
docker images
查詢結果,如圖所示。
這裏,對幾個名詞解釋一下含義。
- REPOSITORY:倉庫名稱。
- TAG: 鏡像標簽,其中 lastest 表示最新版本。註意的是,一個鏡像可以有多個標簽,那麽我們就可以通過標簽來管理有用的版本和功能標簽。
- IMAGE ID :鏡像唯一ID。
- CREATED :創建時間。
- SIZE :鏡像大小。
那麽,如果第一次我們通過 docker pull centos:latest
?拉取鏡像,那麽當我們執行?docker run -p 80 --name web -i -t centos /bin/bash
?時,它就不會再去遠程獲取了,因為本機中已經安裝該鏡像,所以 Docker 會直接創建一個新的容器並啟動該程序。
事實上,官方已經提供了安裝好 Nginx 的鏡像,我們可以直接使用。現在,我們通過拉取鏡像的方式重新構建一個 Web 服務器。首先,我們通過 docker search
?來查找鏡像。我們獲取到 Nginx 的鏡像清單。
docker search nginx
補充一下,我們也可以通過訪問 Docker Hub (https://hub.docker.com/)搜索倉庫,那麽 star 數越多,說明它越靠譜,可以放心使用。
現在,我們通過 docker pull nginx
?拉取最新的?Nginx 的鏡像。當然,我們也可以通過?docker pull nginx:latest
? 來操作。
docker pull nginx
然後,我們創建並運行一個容器。與前面不同的是,我們通過 -d
?選項告訴 Docker 在後臺運行容器的守護進程。並且,通過 8080:80
?告訴 Docker 8080 端口是對外開放的端口,80 端口對外開放的端口映射到容器裏的端口號。
docker run -p 8080:80 -d --name nginx nginx
我們再通過?docker ps -a
?來查看,發現容器已經後臺運行了,並且後臺執行了 nginx 命令,並對外開放 8080 端口。
因此,通過瀏覽器訪問?http://127.0.0.1:8080
?即可。
6. 其他選擇,使用替代註冊服務器
Docker Hub 不是軟件的唯一來源,我們也可以切換到國內的其他替代註冊服務器,例如阿裏雲。我們可以登錄?https://cr.console.aliyun.com?搜索,並拉取公開的鏡像。
現在,我們輸入 docker pull
?命令進行拉取。
docker pull registry.cn-hangzhou.aliyuncs.com/qp_oraclejava/orackejava:8u172_DCEVM_HOTSWAPAGEN_JCE
這裏,筆者繼續補充一個知識點:註冊服務器的地址。事實上,註冊服務器的地址是有一套規範的。完整格式是:[倉庫主機/][用戶名/]容器短名[:標簽]。這裏,倉庫主機是?registry.cn-hangzhou.aliyuncs.com,用戶名是?qp_oraclejava,容器短名是?orackejava,標簽名是 8u172_DCEVM_HOTSWAPAGEN_JCE。事實上,我們上面通過?docker pull centos:latest
?拉取鏡像,相當於?docker pull registry.hub.docker.com/centos:latest
?。
三、構建我的鏡像
通過上面的學習,筆者相信你已經對 Docker 使用有了一個大致的了解,就好比我們通過 VMware 安裝了一個系統,並讓它跑了起來,那麽我們就可以在這個 Linux 系統(CentOS 或者 Ubuntu ) 上面工作我們想要的任何事情。事實上,我們還會經常把我們安裝好的 VMware 系統進行快照備份並實現克隆來滿足我們下次快速的復制。這裏,Docker 也可以構建定制內容的 Docker 鏡像,例如上面我們使用官方提供的安裝好 Nginx 的 Docker 鏡像。註意的是,我們通過基於已有的基礎鏡像,在上面添加鏡像層的方式構建新鏡像而已。
總結一下,Docker 提供自定義鏡像的能力,它可以讓我們保存對基礎鏡像的修改,並再次使用。那麽,我們就可以把操作系統、運行環境、腳本和程序打包在一起,並在宿主機上對外提供服務。
Docker 構建鏡像有兩種方式,一種方式是使用?docker commit
?命令,另外一種方式使用 docker build
?命令和 Dockerfile
?文件。其中,不推薦使用?docker commit
?命令進行構建,因為它沒有使得整個流程標準化,因此,在企業的中更加推薦使用?docker build
?命令和?Dockerfile
?文件來構建我們的鏡像。我們使用Dockerfile
?文件可以讓構建鏡像更具備可重復性,同時保證啟動腳本和運行程序的標準化。
1. 構建第一個 Dockerfile 文件
現在,我們繼續實戰。這裏,我們把一開始搭建的 Web 服務器構建一個鏡像。首先,我們需要創建一個空的 Dokcerfile 文件。
mkdir dockerfile_test
cd dockerfile_test/
touch Dockerfile
nano Dockerfile
緊接著,我們需要編寫一個 Dockerfile 文件,代碼清單如下
FROM centos:7
MAINTAINER LiangGzone "[email protected]"
RUN rpm -ivh http://nginx.org/packages/centos/7/noarch/RPMS/nginx-release-centos-7-0.el7.ngx.noarch.rpm
RUN yum install -y nginx
EXPOSE 80
最後,我們通過 docker build
?命令進行構建。
docker build -t="lianggzone/nginx_demo:v1" .
現在, 我們來通過?docker images
?看下我們的新鏡像吧。
2. 理解 Dockerfile 全過程
哇,我們通過編寫一個 Dockerfile 文件順利構建了一個新的鏡像。這個過程簡單得讓人無法相信。現在,讓我們來理解一下這個全過程吧。首先,?FROM centos:7
?是 Dockerfile 必須要的第一步,它會從一個已經存在的鏡像運行一個容器,換句話說,Docker 需要依賴於一個基礎鏡像進行構建。這裏,我們指定 centos 作為基礎鏡像,它的版本是 7 (CentOS 7)。然後,我們通過?MAINTAINER LiangGzone "[email protected]"
?指定該鏡像的作者是?LiangGzone,郵箱是[email protected]。這有助於告訴使用者它的作者和聯系方式。接著,我們執行兩個 RUN 指令進行 Nginx 的下載安裝,最後通過??EXPOSE 80
?暴露 Dokcer 容器的 80 端口。註意的是,Docker 的執行順序是從上而下執行的,所以我們要明確整個流程的執行順序。除此之外,Docker 在執行每個指令之後都會創建一個新的鏡像層並且進行提交。
我們使用??docker build
?命令進行構建,指定 - t
?告訴 Docker 鏡像的名稱和版本。註意的是,如果沒有指定任何標簽,Docker 將會自動為鏡像設置一個 lastest 標簽。還有一點,我們最後還有一個 .
?是為了讓 Docker 到當前本地目錄去尋找 Dockerfile 文件。註意的是,Docker 會在每一步構建都會將結果提交為鏡像,然後將之前的鏡像層看作緩存,因此我們重新構建類似的鏡像層時會直接復用之前的鏡像。如果我們需要跳過,可以使用?--no-cache
?選項告訴 Docker 不進行緩存。
3. Dockerfile 指令詳解
Dockerfile 提供了非常多的指令。筆者這裏特別整理了一份清單,建議收藏查看。
官方地址:https://docs.docker.com/engine/reference/builder/#usage
指令辨別一:RUN、CMD、ENTRYPOINT
RUN
?、 CMD
?、 ENTRYPOINT
? 三個指令的用途非常相識,不同在於,RUN
?指令是在容器被構建時運行的命令,而CMD
?、?ENTRYPOINT
?是啟動容器時執行 shell 命令,而?RUN
?會被 docker run
?命令覆蓋,但是??ENTRYPOINT
?不會被覆蓋。事實上,docker run
?命令指定的任何參數都會被當作參數再次傳遞給?ENTRYPOINT
? 指令。CMD
?、?ENTRYPOINT
?兩個指令之間也可以一起使用。例如,我們 可以使用?ENTRYPOINT
?的 exec 形式設置固定的默認命令和參數,然後使用任一形式的 CMD
?來設置可能更改的其他默認值。
FROM ubuntu
ENTRYPOINT ["top", "-b"]
CMD ["-c"]
指令辨別二:ADD、COPY
ADD
?、 COPY
?指令用法一樣,唯一不同的是?ADD
? 支持將歸檔文件(tar, gzip, bzip2, etc)做提取和解壓操作。註意的是,COPY
?指令需要復制的目錄一定要放在 Dockerfile 文件的同級目錄下。
4. 將鏡像推送到遠程倉庫
遠程倉庫:Docker Hub?
鏡像構建完畢之後,我們可以將它上傳到 Docker Hub 上面。首先,我們需要通過 docker login
?保證我們已經登錄了。緊接著,我們使用 docker push
?命令進行推送。
docker push lianggzone/nginx_demo:v1
這裏,我們了解下它的使用,格式是 docker push [OPTIONS] NAME[:TAG]
?,其中,筆者設置 NAME 是?lianggzone/nginx_demo,TAG 是 v1。 (筆者註:推送 Docker Hub 速度很慢,耐心等待) 最後,上傳完成後訪問:https://hub.docker.com/u/lianggzone/,如圖所示。
遠程倉庫:阿裏雲
同時,我們也可以使用國內的倉庫,比如阿裏雲。首先,在終端中輸入訪問憑證,登錄 Registry 實例。如果你不知道是哪個,可以訪問?https://cr.console.aliyun.com/cn-hangzhou/instances/credentials。
docker login --username=帳號 registry.cn-hangzhou.aliyuncs.com
現在,將鏡像推送到阿裏雲鏡像倉庫。其中, docker tag [IMAGE_ID] registry.cn-hangzhou.aliyuncs.com/[命名空間]/[鏡像名稱]:[版本]
?和 docker push registry.cn-hangzhou.aliyuncs.com/[命名空間]/[鏡像名稱]:[版本]
?命令的使用如下所示。
docker tag 794c07361565 registry.cn-hangzhou.aliyuncs.com/lianggzone/nginx_demo:v1
docker push registry.cn-hangzhou.aliyuncs.com/lianggzone/nginx_demo:v1
最後,上傳完成後訪問:https://cr.console.aliyun.com/cn-hangzhou/instances/repositories,如圖所示。
5.?Dockerfile 的 Github 源碼地址
這裏,附上我整理的 Dockerfile 的倉庫。後面,筆者會陸續更新用到的一些常用文件,歡迎 star 關註。
https://github.com/lianggzone/dockerfile-images
附:參考資料
- 《Docker實戰》
- 《第一本Docker書》
- Docker 命令參考文檔
- Dockerfile 鏡像構建參考文檔
(完,轉載請註明作者及出處。)
寫在末尾
【服務端思維】:我們一起聊聊服務端核心技術,探討一線互聯網的項目架構與實戰經驗。同時,擁有眾多技術大牛的「後端圈」大家庭,期待你的加入,一群同頻者,一起成長,一起精進,打破認知的局限性。
更多精彩文章,盡在「服務端思維」!
30 分鐘快速入門 Docker 教程