2013年發(fā)布至今, Docker 一直廣受矚目,被認(rèn)為可能會改變軟件行業(yè)。
但是,許多人并不清楚 Docker 到底是什么,要解決什么問題,好處又在哪里?本文就來詳細(xì)解釋,幫助大家理解它,還帶有簡單易懂的實例,教你如何將它用于日常開發(fā)。
一、環(huán)境配置的難題
軟件開發(fā)最大的麻煩事之一,就是環(huán)境配置。用戶計算機(jī)的環(huán)境都不相同,你怎么知道自家的軟件,能在那些機(jī)器跑起來?
用戶必須保證兩件事:操作系統(tǒng)的設(shè)置,各種庫和組件的安裝。只有它們都正確,軟件才能運行。舉例來說,安裝一個 Python/ target=_blank class=infotextkey>Python 應(yīng)用,計算機(jī)必須有 Python 引擎,還必須有各種依賴,可能還要配置環(huán)境變量。
如果某些老舊的模塊與當(dāng)前環(huán)境不兼容,那就麻煩了。開發(fā)者常常會說:"它在我的機(jī)器可以跑了"(It works on my machine),言下之意就是,其他機(jī)器很可能跑不了。
環(huán)境配置如此麻煩,換一臺機(jī)器,就要重來一次,曠日費時。很多人想到,能不能從根本上解決問題,軟件可以帶環(huán)境安裝?也就是說,安裝的時候,把原始環(huán)境一模一樣地復(fù)制過來。
二、虛擬機(jī)
虛擬機(jī)(virtual machine)就是帶環(huán)境安裝的一種解決方案。它可以在一種操作系統(tǒng)里面運行另一種操作系統(tǒng),比如在 windows 系統(tǒng)里面運行 linux 系統(tǒng)。應(yīng)用程序?qū)Υ撕翢o感知,因為虛擬機(jī)看上去跟真實系統(tǒng)一模一樣,而對于底層系統(tǒng)來說,虛擬機(jī)就是一個普通文件,不需要了就刪掉,對其他部分毫無影響。
雖然用戶可以通過虛擬機(jī)還原軟件的原始環(huán)境。但是,這個方案有幾個缺點。
(1)資源占用多
虛擬機(jī)會獨占一部分內(nèi)存和硬盤空間。它運行的時候,其他程序就不能使用這些資源了。哪怕虛擬機(jī)里面的應(yīng)用程序,真正使用的內(nèi)存只有 1MB,虛擬機(jī)依然需要幾百 MB 的內(nèi)存才能運行。
(2)冗余步驟多
虛擬機(jī)是完整的操作系統(tǒng),一些系統(tǒng)級別的操作步驟,往往無法跳過,比如用戶登錄。
(3)啟動慢
啟動操作系統(tǒng)需要多久,啟動虛擬機(jī)就需要多久??赡芤葞追昼姡瑧?yīng)用程序才能真正運行。
三、Linux 容器
由于虛擬機(jī)存在這些缺點,Linux 發(fā)展出了另一種虛擬化技術(shù):Linux 容器(Linux Containers,縮寫為 LXC)。
Linux 容器不是模擬一個完整的操作系統(tǒng),而是對進(jìn)程進(jìn)行隔離?;蛘哒f,在正常進(jìn)程的外面套了一個保護(hù)層。對于容器里面的進(jìn)程來說,它接觸到的各種資源都是虛擬的,從而實現(xiàn)與底層系統(tǒng)的隔離。
由于容器是進(jìn)程級別的,相比虛擬機(jī)有很多優(yōu)勢。
(1)啟動快
容器里面的應(yīng)用,直接就是底層系統(tǒng)的一個進(jìn)程,而不是虛擬機(jī)內(nèi)部的進(jìn)程。所以,啟動容器相當(dāng)于啟動本機(jī)的一個進(jìn)程,而不是啟動一個操作系統(tǒng),速度就快很多。
(2)資源占用少
容器只占用需要的資源,不占用那些沒有用到的資源;虛擬機(jī)由于是完整的操作系統(tǒng),不可避免要占用所有資源。另外,多個容器可以共享資源,虛擬機(jī)都是獨享資源。
(3)體積小
容器只要包含用到的組件即可,而虛擬機(jī)是整個操作系統(tǒng)的打包,所以容器文件比虛擬機(jī)文件要小很多。
總之,容器有點像輕量級的虛擬機(jī),能夠提供虛擬化的環(huán)境,但是成本開銷小得多。
四、Docker 是什么?
docker 屬于 Linux 容器的一種封裝,提供簡單易用的容器使用接口。它是目前最流行的 Linux 容器解決方案。
Docker 將應(yīng)用程序與該程序的依賴,打包在一個文件里面。運行這個文件,就會生成一個虛擬容器。程序在這個虛擬容器里運行,就好像在真實的物理機(jī)上運行一樣。有了 Docker,就不用擔(dān)心環(huán)境問題。
總體來說,Docker 的接口相當(dāng)簡單,用戶可以方便地創(chuàng)建和使用容器,把自己的應(yīng)用放入容器。容器還可以進(jìn)行版本管理、復(fù)制、分享、修改,就像管理普通的代碼一樣。
五、Docker 的用途
Docker 的主要用途,目前有三大類。
(1)提供一次性的環(huán)境。比如,本地測試他人的軟件、持續(xù)集成的時候提供單元測試和構(gòu)建的環(huán)境。
(2)提供彈性的云服務(wù)。因為 Docker 容器可以隨開隨關(guān),很適合動態(tài)擴(kuò)容和縮容。
(3)組建微服務(wù)架構(gòu)。通過多個容器,一臺機(jī)器可以跑多個服務(wù),因此在本機(jī)就可以模擬出微服務(wù)架構(gòu)。
六、Docker 的安裝
Docker 是一個開源的商業(yè)產(chǎn)品,有兩個版本:社區(qū)版(Community Edition,縮寫為 CE)和企業(yè)版(Enterprise Edition,縮寫為 EE)。企業(yè)版包含了一些收費服務(wù),個人開發(fā)者一般用不到。下面的介紹都針對社區(qū)版。
Docker CE 的安裝請參考官方文檔。
Mac Windows Ubuntu Debian centos Fedora 其他 Linux 發(fā)行版
安裝完成后,運行下面的命令,驗證是否安裝成功。
$ docker version # 或者 $ docker info
Docker 需要用戶具有 sudo 權(quán)限,為了避免每次命令都輸入sudo,可以把用戶加入 Docker 用戶組。
$ sudo usermod -aG docker $USER
Docker 是服務(wù)器----客戶端架構(gòu)。命令行運行docker命令的時候,需要本機(jī)有 Docker 服務(wù)。如果這項服務(wù)沒有啟動,可以用下面的命令啟動。
# service 命令的用法 $ sudo service docker start # systemctl 命令的用法 $ sudo systemctl start docker六、image 文件
Docker 把應(yīng)用程序及其依賴,打包在 image 文件里面。只有通過這個文件,才能生成 Docker 容器。image 文件可以看作是容器的模板。Docker 根據(jù) image 文件生成容器的實例。同一個 image 文件,可以生成多個同時運行的容器實例。
image 是二進(jìn)制文件。實際開發(fā)中,一個 image 文件往往通過繼承另一個 image 文件,加上一些個性化設(shè)置而生成。舉例來說,你可以在 Ubuntu 的 image 基礎(chǔ)上,往里面加入 Apache 服務(wù)器,形成你的 image。
# 列出本機(jī)的所有 image 文件。 $ docker image ls # 刪除 image 文件 $ docker image rm [imageName]
image 文件是通用的,一臺機(jī)器的 image 文件拷貝到另一臺機(jī)器,照樣可以使用。一般來說,為了節(jié)省時間,我們應(yīng)該盡量使用別人制作好的 image 文件,而不是自己制作。即使要定制,也應(yīng)該基于別人的 image 文件進(jìn)行加工,而不是從零開始制作。
為了方便共享,image 文件制作完成后,可以上傳到網(wǎng)上的倉庫。Docker 的官方倉庫 Docker Hub 是最重要、最常用的 image 倉庫。此外,出售自己制作的 image 文件也是可以的。
七、實例:hello world
下面,我們通過最簡單的 image 文件"hello world",感受一下 Docker。
首先,運行下面的命令,將 image 文件從倉庫抓取到本地。
$ docker image pull library/hello-world
上面代碼中,docker image pull是抓取 image 文件的命令。library/hello-world是 image 文件在倉庫里面的位置,其中l(wèi)ibrary是 image 文件所在的組,hello-world是 image 文件的名字。
由于 Docker 官方提供的 image 文件,都放在library組里面,所以它的是默認(rèn)組,可以省略。因此,上面的命令可以寫成下面這樣。
$ docker image pull hello-world
抓取成功以后,就可以在本機(jī)看到這個 image 文件了。
$ docker image ls
現(xiàn)在,運行這個 image 文件。
$ docker container run hello-world
docker container run命令會從 image 文件,生成一個正在運行的容器實例。
注意,docker container run命令具有自動抓取 image 文件的功能。如果發(fā)現(xiàn)本地沒有指定的 image 文件,就會從倉庫自動抓取。因此,前面的docker image pull命令并不是必需的步驟。
如果運行成功,你會在屏幕上讀到下面的輸出。
$ docker container run hello-world Hello from Docker! This message shows that your installation Appears to be working correctly. ... ...
輸出這段提示以后,hello world就會停止運行,容器自動終止。
有些容器不會自動終止,因為提供的是服務(wù)。比如,安裝運行 Ubuntu 的 image,就可以在命令行體驗 Ubuntu 系統(tǒng)。
$ docker container run -it ubuntu Bash
對于那些不會自動終止的容器,必須使用docker container kill 命令手動終止。
$ docker container kill [containID]八、容器文件
image 文件生成的容器實例,本身也是一個文件,稱為容器文件。也就是說,一旦容器生成,就會同時存在兩個文件:image 文件和容器文件。而且關(guān)閉容器并不會刪除容器文件,只是容器停止運行而已。
# 列出本機(jī)正在運行的容器 $ docker container ls # 列出本機(jī)所有容器,包括終止運行的容器 $ docker container ls --all
上面命令的輸出結(jié)果之中,包括容器的 ID。很多地方都需要提供這個 ID,比如上一節(jié)終止容器運行的docker container kill命令。
終止運行的容器文件,依然會占據(jù)硬盤空間,可以使用docker container rm命令刪除。
$ docker container rm [containerID]
運行上面的命令之后,再使用docker container ls --all命令,就會發(fā)現(xiàn)被刪除的容器文件已經(jīng)消失了。
九、Dockerfile 文件
學(xué)會使用 image 文件以后,接下來的問題就是,如何可以生成 image 文件?如果你要推廣自己的軟件,勢必要自己制作 image 文件。
這就需要用到 Dockerfile 文件。它是一個文本文件,用來配置 image。Docker 根據(jù) 該文件生成二進(jìn)制的 image 文件。
下面通過一個實例,演示如何編寫 Dockerfile 文件。
十、實例:制作自己的 Docker 容器
下面我以 koa-demos 項目為例,介紹怎么寫 Dockerfile 文件,實現(xiàn)讓用戶在 Docker 容器里面運行 Koa 框架。
作為準(zhǔn)備工作,請先下載源碼。
$ git clone https://github.com/ruanyf/koa-demos.git $ cd koa-demos編寫 Dockerfile 文件
首先,在項目的根目錄下,新建一個文本文件.dockerignore,寫入下面的內(nèi)容。
.git node_modules npm-debug.log
上面代碼表示,這三個路徑要排除,不要打包進(jìn)入 image 文件。如果你沒有路徑要排除,這個文件可以不新建。
然后,在項目的根目錄下,新建一個文本文件 Dockerfile,寫入下面的內(nèi)容。
FROM node:8.4 COPY . /app WORKDIR /app RUN npm install --registry=https://registry.npm.taobao.org EXPOSE 3000
上面代碼一共五行,含義如下。
FROM node:8.4:該 image 文件繼承官方的 node image,冒號表示標(biāo)簽,這里標(biāo)簽是8.4,即8.4版本的 node。 COPY . /app:將當(dāng)前目錄下的所有文件(除了.dockerignore排除的路徑),都拷貝進(jìn)入 image 文件的/app目錄。 WORKDIR /app:指定接下來的工作路徑為/app。 RUN npm install:在/app目錄下,運行npm install命令安裝依賴。注意,安裝后所有的依賴,都將打包進(jìn)入 image 文件。 EXPOSE 3000:將容器 3000 端口暴露出來, 允許外部連接這個端口。創(chuàng)建 image 文件
有了 Dockerfile 文件以后,就可以使用docker image build命令創(chuàng)建 image 文件了。
$ docker image build -t koa-demo . # 或者 $ docker image build -t koa-demo:0.0.1 .
上面代碼中,-t參數(shù)用來指定 image 文件的名字,后面還可以用冒號指定標(biāo)簽。如果不指定,默認(rèn)的標(biāo)簽就是latest。最后的那個點表示 Dockerfile 文件所在的路徑,上例是當(dāng)前路徑,所以是一個點。
如果運行成功,就可以看到新生成的 image 文件koa-demo了。
$ docker image ls生成容器
docker container run命令會從 image 文件生成容器。
$ docker container run -p 8000:3000 -it koa-demo /bin/bash # 或者 $ docker container run -p 8000:3000 -it koa-demo:0.0.1 /bin/bash
上面命令的各個參數(shù)含義如下:
-p參數(shù):容器的 3000 端口映射到本機(jī)的 8000 端口。 -it參數(shù):容器的 Shell 映射到當(dāng)前的 Shell,然后你在本機(jī)窗口輸入的命令,就會傳入容器。 koa-demo:0.0.1:image 文件的名字(如果有標(biāo)簽,還需要提供標(biāo)簽,默認(rèn)是 latest 標(biāo)簽)。 /bin/bash:容器啟動以后,內(nèi)部第一個執(zhí)行的命令。這里是啟動 Bash,保證用戶可以使用 Shell。
如果一切正常,運行上面的命令以后,就會返回一個命令行提示符。
root@66d80f4aaf1e:/app#
這表示你已經(jīng)在容器里面了,返回的提示符就是容器內(nèi)部的 Shell 提示符。執(zhí)行下面的命令。
root@66d80f4aaf1e:/app# node demos/01.js
這時,Koa 框架已經(jīng)運行起來了。打開本機(jī)的瀏覽器,訪問 http://127.0.0.1:8000,網(wǎng)頁顯示"Not Found",這是因為這個 demo 沒有寫路由。
這個例子中,Node 進(jìn)程運行在 Docker 容器的虛擬環(huán)境里面,進(jìn)程接觸到的文件系統(tǒng)和網(wǎng)絡(luò)接口都是虛擬的,與本機(jī)的文件系統(tǒng)和網(wǎng)絡(luò)接口是隔離的,因此需要定義容器與物理機(jī)的端口映射(map)。
現(xiàn)在,在容器的命令行,按下 Ctrl + c 停止 Node 進(jìn)程,然后按下 Ctrl + d (或者輸入 exit)退出容器。此外,也可以用docker container kill終止容器運行。
# 在本機(jī)的另一個終端窗口,查出容器的 ID $ docker container ls # 停止指定的容器運行 $ docker container kill [containerID]
容器停止運行之后,并不會消失,用下面的命令刪除容器文件。
# 查出容器的 ID $ docker container ls --all # 刪除指定的容器文件 $ docker container rm [containerID]
也可以使用docker container run命令的--rm參數(shù),在容器終止運行后自動刪除容器文件。
$ docker container run --rm -p 8000:3000 -it koa-demo /bin/bashCMD 命令
上一節(jié)的例子里面,容器啟動以后,需要手動輸入命令node demos/01.js。我們可以把這個命令寫在 Dockerfile 里面,這樣容器啟動以后,這個命令就已經(jīng)執(zhí)行了,不用再手動輸入了。
FROM node:8.4 COPY . /app WORKDIR /app RUN npm install --registry=https://registry.npm.taobao.org EXPOSE 3000 CMD node demos/01.js
上面的 Dockerfile 里面,多了最后一行CMD node demos/01.js,它表示容器啟動后自動執(zhí)行node demos/01.js。
你可能會問,RUN命令與CMD命令的區(qū)別在哪里?簡單說,RUN命令在 image 文件的構(gòu)建階段執(zhí)行,執(zhí)行結(jié)果都會打包進(jìn)入 image 文件;CMD命令則是在容器啟動后執(zhí)行。另外,一個 Dockerfile 可以包含多個RUN命令,但是只能有一個CMD命令。
注意,指定了CMD命令以后,docker container run命令就不能附加命令了(比如前面的/bin/bash),否則它會覆蓋CMD命令?,F(xiàn)在,啟動容器可以使用下面的命令。
$ docker container run --rm -p 8000:3000 -it koa-demo:0.0.1發(fā)布 image 文件
容器運行成功后,就確認(rèn)了 image 文件的有效性。這時,我們就可以考慮把 image 文件分享到網(wǎng)上,讓其他人使用。
首先,去 hub.docker.com 或 cloud.docker.com 注冊一個賬戶。然后,用下面的命令登錄。
$ docker login
接著,為本地的 image 標(biāo)注用戶名和版本。
$ docker image tag [imageName] [username]/[repository]:[tag] # 實例 $ docker image tag koa-demos:0.0.1 ruanyf/koa-demos:0.0.1
也可以不標(biāo)注用戶名,重新構(gòu)建一下 image 文件。
$ docker image build -t [username]/[repository]:[tag] .
最后,發(fā)布 image 文件。
$ docker image push [username]/[repository]:[tag]
發(fā)布成功以后,登錄 hub.docker.com,就可以看到已經(jīng)發(fā)布的 image 文件。
十一、其他有用的命令
docker 的主要用法就是上面這些,此外還有幾個命令,也非常有用。
(1)docker container start
前面的docker container run命令是新建容器,每運行一次,就會新建一個容器。同樣的命令運行兩次,就會生成兩個一模一樣的容器文件。如果希望重復(fù)使用容器,就要使用docker container start命令,它用來啟動已經(jīng)生成、已經(jīng)停止運行的容器文件。
$ docker container start [containerID]
(2)docker container stop
前面的docker container kill命令終止容器運行,相當(dāng)于向容器里面的主進(jìn)程發(fā)出 SIGKILL 信號。而docker container stop命令也是用來終止容器運行,相當(dāng)于向容器里面的主進(jìn)程發(fā)出 SIGTERM 信號,然后過一段時間再發(fā)出 SIGKILL 信號。
$ docker container stop [containerID]
這兩個信號的差別是,應(yīng)用程序收到 SIGTERM 信號以后,可以自行進(jìn)行收尾清理工作,但也可以不理會這個信號。如果收到 SIGKILL 信號,就會強(qiáng)行立即終止,那些正在進(jìn)行中的操作會全部丟失。
(3)docker container logs
docker container logs命令用來查看 docker 容器的輸出,即容器里面 Shell 的標(biāo)準(zhǔn)輸出。如果docker run命令運行容器的時候,沒有使用-it參數(shù),就要用這個命令查看輸出。
$ docker container logs [containerID]
(4)docker container exec
docker container exec命令用于進(jìn)入一個正在運行的 docker 容器。如果docker run命令運行容器的時候,沒有使用-it參數(shù),就要用這個命令進(jìn)入容器。一旦進(jìn)入了容器,就可以在容器的 Shell 執(zhí)行命令了。
$ docker container exec -it [containerID] /bin/bash
(5)docker container cp
docker container cp命令用于從正在運行的 Docker 容器里面,將文件拷貝到本機(jī)。下面是拷貝到當(dāng)前目錄的寫法。
$ docker container cp [containID]:[/path/to/file] .
Docker 是一個容器工具,提供虛擬環(huán)境。很多人認(rèn)為,它改變了我們對軟件的認(rèn)識。
站在 Docker 的角度,軟件就是容器的組合:業(yè)務(wù)邏輯容器、數(shù)據(jù)庫容器、儲存容器、隊列容器......Docker 使得軟件可以拆分成若干個標(biāo)準(zhǔn)化容器,然后像搭積木一樣組合起來。
這正是微服務(wù)(microservices)的思想:軟件把任務(wù)外包出去,讓各種外部服務(wù)完成這些任務(wù),軟件本身只是底層服務(wù)的調(diào)度中心和組裝層。
微服務(wù)很適合用 Docker 容器實現(xiàn),每個容器承載一個服務(wù)。一臺計算機(jī)同時運行多個容器,從而就能很輕松地模擬出復(fù)雜的微服務(wù)架構(gòu)。
下面我們介紹一下如何在一臺計算機(jī)上實現(xiàn)多個服務(wù),讓它們互相配合,組合出一個應(yīng)用程序。
我選擇的示例軟件是 wordPress/ target=_blank class=infotextkey>WordPress。它是一個常用軟件,全世界用戶據(jù)說超過幾千萬。同時它又非常簡單,只要兩個容器就夠了(業(yè)務(wù)容器 + 數(shù)據(jù)庫容器),很適合教學(xué)。而且,這種"業(yè)務(wù) + 數(shù)據(jù)庫"的容器架構(gòu),具有通用性,許多應(yīng)用程序都可以復(fù)用。
為了加深讀者理解,本文采用三種方法,演示如何架設(shè) WordPress 網(wǎng)站。
方法 A:自建 WordPress 容器 方法 B:采用官方的 WordPress 容器 方法 C:采用 Docker Compose 工具十二、預(yù)備工作:image 倉庫的鏡像網(wǎng)址
本教程需要從倉庫下載 image 文件,但是國內(nèi)訪問 Docker 的官方倉庫很慢,還經(jīng)常斷線,所以要把倉庫網(wǎng)址改成國內(nèi)的鏡像站。這里推薦使用官方鏡像 registry.docker-cn.com 。下面是我的 Debian 系統(tǒng)的默認(rèn)倉庫修改方法,其他系統(tǒng)的修改方法參考官方文檔。
打開/etc/default/docker文件(需要sudo權(quán)限),在文件的底部加上一行。
DOCKER_OPTS="--registry-mirror=https://registry.docker-cn.com"
然后,重啟 Docker 服務(wù)。
$ sudo service docker restart
現(xiàn)在就會自動從鏡像倉庫下載 image 文件了。
十三、方法 A:自建 WordPress 容器
前面說過,本文會用三種方法演示 WordPress 的安裝。第一種方法就是自建 WordPress 容器。
官方 的 php image
首先,新建一個工作目錄,并進(jìn)入該目錄。
$ mkdir docker-demo && cd docker-demo
然后,執(zhí)行下面的命令。
$ docker container run --rm --name wordpress --volume "$PWD/":/var/www/html php:5.6-apache
上面的命令基于php的 image 文件新建一個容器,并且運行該容器。php的標(biāo)簽是5.6-apache,說明裝的是 PHP 5.6,并且自帶 Apache 服務(wù)器。該命令的三個參數(shù)含義如下。
--rm:停止運行后,自動刪除容器文件。 --name wordpress:容器的名字叫做wordpress。 --volume "$PWD/":/var/www/html:將當(dāng)前目錄($PWD)映射到容器的/var/www/html(Apache 對外訪問的默認(rèn)目錄)。因此,當(dāng)前目錄的任何修改,都會反映到容器里面,進(jìn)而被外部訪問到。
運行上面的命令以后,如果一切正常,命令行會提示容器對外的 IP 地址,請記下這個地址,我們要用它來訪問容器。我分配到的 IP 地址是 172.17.0.2。
打開瀏覽器,訪問 172.17.0.2,你會看到下面的提示。
Forbidden You don't have permission to access / on this server.
這是因為容器的/var/www/html目錄(也就是本機(jī)的docker-demo目錄)下面什么也沒有,無法提供可以訪問的內(nèi)容。
請在本機(jī)的docker-demo目錄下面,添加一個最簡單的 PHP 文件index.php。
phpinfo(); ?>
保存以后,瀏覽器刷新172.17.0.2,應(yīng)該就會看到熟悉的phpinfo頁面了。
拷貝 WordPress 安裝包
既然本地的docker-demo目錄可以映射到容器里面,那么把 WordPress 安裝包拷貝到docker-demo目錄下,不就可以通過容器訪問到 WordPress 的安裝界面了嗎?
首先,在docker-demo目錄下,執(zhí)行下面的命令,抓取并解壓 WordPress 安裝包。
$ wget https://cn.wordpress.org/wordpress-4.9.4-zh_CN.tar.gz $ tar -xvf wordpress-4.9.4-zh_CN.tar.gz
解壓以后,WordPress 的安裝文件會在docker-demo/wordpress目錄下。
這時瀏覽器訪問http://172.17.0.2/wordpress,就能看到 WordPress 的安裝提示了。
官方的 MySQL 容器
WordPress 必須有數(shù)據(jù)庫才能安裝,所以必須新建 MySQL 容器。
打開一個新的命令行窗口,執(zhí)行下面的命令。
$ docker container run -d --rm --name wordpressdb --env MYSQL_ROOT_PASSWORD=123456 --env MYSQL_DATABASE=wordpress mysql:5.7
上面的命令會基于 MySQL 的 image 文件(5.7版本)新建一個容器。該命令的五個命令行參數(shù)的含義如下。
-d:容器啟動后,在后臺運行。 --rm:容器終止運行后,自動刪除容器文件。 --name wordpressdb:容器的名字叫做wordpressdb --env MYSQL_ROOT_PASSWORD=123456:向容器進(jìn)程傳入一個環(huán)境變量MYSQL_ROOT_PASSWORD,該變量會被用作 MySQL 的根密碼。 --env MYSQL_DATABASE=wordpress:向容器進(jìn)程傳入一個環(huán)境變量MYSQL_DATABASE,容器里面的 MySQL 會根據(jù)該變量創(chuàng)建一個同名數(shù)據(jù)庫(本例是WordPress)。
運行上面的命令以后,正常情況下,命令行會顯示一行字符串,這是容器的 ID,表示已經(jīng)新建成功了。
這時,使用下面的命令查看正在運行的容器,你應(yīng)該看到wordpress和wordpressdb兩個容器正在運行。
$ docker container ls
其中,wordpressdb是后臺運行的,前臺看不見它的輸出,必須使用下面的命令查看。
$ docker container logs wordpressdb定制 PHP 容器
現(xiàn)在 WordPress 容器和 MySQL 容器都已經(jīng)有了。接下來,要把 WordPress 容器連接到 MySQL 容器了。但是,PHP 的官方 image 不帶有mysql擴(kuò)展,必須自己新建 image 文件。
首先,停掉 WordPress 容器。
$ docker container stop wordpress
停掉以后,由于--rm參數(shù)的作用,該容器文件會被自動刪除。
然后,在docker-demo目錄里面,新建一個Dockerfile文件,寫入下面的內(nèi)容。
FROM php:5.6-apache RUN docker-php-ext-install mysqli CMD apache2-foreground
上面代碼的意思,就是在原來 PHP 的 image 基礎(chǔ)上,安裝mysqli的擴(kuò)展。然后,啟動 Apache。
基于這個 Dockerfile 文件,新建一個名為phpwithmysql的 image 文件。
$ docker build -t phpwithmysql .Wordpress 容器連接 MySQL
現(xiàn)在基于 phpwithmysql image,重新新建一個 WordPress 容器。
$ docker container run --rm --name wordpress --volume "$PWD/":/var/www/html --link wordpressdb:mysql phpwithmysql
跟上一次相比,上面的命令多了一個參數(shù)--link wordpressdb:mysql,表示 WordPress 容器要連到wordpressdb容器,冒號表示該容器的別名是mysql。
這時還要改一下wordpress目錄的權(quán)限,讓容器可以將配置信息寫入這個目錄(容器內(nèi)部寫入的/var/www/html目錄,會映射到這個目錄)。
$ chmod -R 777 wordpress
接著,回到瀏覽器的http://172.17.0.2/wordpress頁面,點擊"現(xiàn)在就開始!"按鈕,開始安裝。
WordPress 提示要輸入數(shù)據(jù)庫參數(shù)。輸入的參數(shù)如下。
數(shù)據(jù)庫名:wordpress 用戶名:root 密碼:123456 數(shù)據(jù)庫主機(jī):mysql 表前綴:wp_(不變)
點擊"下一步"按鈕,如果 Wordpress 連接數(shù)據(jù)庫成功,就會出現(xiàn)下面的頁面,這就表示可以安裝了。
至此,自建 WordPress 容器的演示完畢,可以把正在運行的兩個容器關(guān)閉了(容器文件會自動刪除)。
$ docker container stop wordpress wordpressdb十四、方法 B:Wordpress 官方鏡像
上一部分的自建 WordPress 容器,還是挺麻煩的。其實不用這么麻煩,Docker 已經(jīng)提供了官方 WordPress image,直接用那個就可以了。有了上一部分的基礎(chǔ),下面的操作就很容易理解了。
基本用法
首先,新建并啟動 MySQL 容器。
$ docker container run -d --rm --name wordpressdb --env MYSQL_ROOT_PASSWORD=123456 --env MYSQL_DATABASE=wordpress mysql:5.7
然后,基于官方的 WordPress image,新建并啟動 WordPress 容器。
$ docker container run -d --rm --name wordpress --env WORDPRESS_DB_PASSWORD=123456 --link wordpressdb:mysql wordpress
上面命令中,各個參數(shù)的含義前面都解釋過了,其中環(huán)境變量WORDPRESS_DB_PASSWORD是 MySQL 容器的根密碼。
上面命令指定wordpress容器在后臺運行,導(dǎo)致前臺看不見輸出,使用下面的命令查出wordpress容器的 IP 地址。
$ docker container inspect wordpress
上面命令運行以后,會輸出很多內(nèi)容,找到IPAddress字段即可。我的機(jī)器返回的 IP 地址是172.17.0.3。
瀏覽器訪問172.17.0.3,就會看到 WordPress 的安裝提示。
WordPress 容器的定制
到了上一步,官方 WordPress 容器的安裝就已經(jīng)成功了。但是,這種方法有兩個很不方便的地方。
每次新建容器,返回的 IP 地址不能保證相同,導(dǎo)致要更換 IP 地址訪問 WordPress。 WordPress 安裝在容器里面,本地?zé)o法修改文件。
解決這兩個問題很容易,只要新建容器的時候,加兩個命令行參數(shù)就可以了。
先把剛才啟動的 WordPress 容器終止(容器文件會自動刪除)。
$ docker container stop wordpress
然后,使用下面的命令新建并啟動 WordPress 容器。
$ docker container run -d -p 127.0.0.2:8080:80 --rm --name wordpress --env WORDPRESS_DB_PASSWORD=123456 --link wordpressdb:mysql --volume "$PWD/wordpress":/var/www/html wordpress
上面的命令跟前面相比,命令行參數(shù)只多出了兩個。
-p 127.0.0.2:8080:80:將容器的 80 端口映射到127.0.0.2的8080端口。 --volume "$PWD/wordpress":/var/www/html:將容器的/var/www/html目錄映射到當(dāng)前目錄的wordpress子目錄。
瀏覽器訪問127.0.0.2:8080:80就能看到 WordPress 的安裝提示了。而且,你在wordpress子目錄下的每次修改,都會反映到容器里面。
最后,終止這兩個容器(容器文件會自動刪除)。
$ docker container stop wordpress wordpressdb十五、方法 C:Docker Compose 工具
上面的方法 B 已經(jīng)挺簡單了,但是必須自己分別啟動兩個容器,啟動的時候,還要在命令行提供容器之間的連接信息。因此,Docker 提供了一種更簡單的方法,來管理多個容器的聯(lián)動。
Docker Compose 簡介
Compose 是 Docker 公司推出的一個工具軟件,可以管理多個 Docker 容器組成一個應(yīng)用。你需要定義一個 YAML 格式的配置文件docker-compose.yml,寫好多個容器之間的調(diào)用關(guān)系。然后,只要一個命令,就能同時啟動/關(guān)閉這些容器。
# 啟動所有服務(wù) $ docker-compose up # 關(guān)閉所有服務(wù) $ docker-compose stopDocker Compose 的安裝
Mac 和 Windows 在安裝 docker 的時候,會一起安裝 docker compose。Linux 系統(tǒng)下的安裝參考官方文檔。
安裝完成后,運行下面的命令。
$ docker-compose --versionWordPress 示例
在docker-demo目錄下,新建docker-compose.yml文件,寫入下面的內(nèi)容。
mysql: image: mysql:5.7 environment: - MYSQL_ROOT_PASSWORD=123456 - MYSQL_DATABASE=wordpress web: image: wordpress links: - mysql environment: - WORDPRESS_DB_PASSWORD=123456 ports: - "127.0.0.3:8080:80" working_dir: /var/www/html volumes: - wordpress:/var/www/html
上面代碼中,兩個頂層標(biāo)簽表示有兩個容器mysql和web。每個容器的具體設(shè)置,前面都已經(jīng)講解過了,還是挺容易理解的。
啟動兩個容器。
$ docker-compose up
瀏覽器訪問 http://127.0.0.3:8080,應(yīng)該就能看到 WordPress 的安裝界面。
現(xiàn)在關(guān)閉兩個容器。
$ docker-compose stop
關(guān)閉以后,這兩個容器文件還是存在的,寫在里面的數(shù)據(jù)不會丟失。下次啟動的時候,還可以復(fù)用。下面的命令可以把這兩個容器文件刪除(容器必須已經(jīng)停止運行)。
$ docker-compose rm