邁出使用Docker的第一步,學習第一個Docker容器
在本文中,我們將邁出使用Docker的第一步,學習第一個Docker容器。本章還會介紹如何與Docker進行交互的基本知識。
1 確保Docker已經就緒
首先,我們會查看Docker是否能正常工作,然后學習基本的Docker的工作流:創建并管理容器。我們將瀏覽容器的典型生命周期:從創建、管理到停止,直到最終刪除。
第一步,查看docker
程序是否存在,功能是否正常,如代碼清單3-1所示。
代碼清單3-1 查看docker
程序是否正常工作
$ sudo docker info Containers: 0 Images: 0 Storage Driver: aufs Root Dir: /var/lib/docker/aufs Dirs: 144 Execution Driver: native-0.1 Kernel Version: 3.8.0-29-generic Registry: [https://index.docker.io/v1/]
在這里我們調用了docker
可執行程序的info
命令,該命令會返回所有容器和鏡像(鏡像即是Docker用來構建容器的“構建塊”)的數量、Docker使用的執行驅動和存儲驅動(execution and storage driver),以及Docker的基本配置。
在前面幾章我們已經介紹過,Docker是基于客戶端-服務器構架的。它有一個docker
程序,既能作為客戶端,也可以作為服務器端。作為客戶端時,docker
程序向Docker守護進程發送請求(如請求返回守護進程自身的信息),然后再對返回來的請求結果進行處理。
2 運行我們的第一個容器
現在,讓我們嘗試啟動第一個Docker容器。我們可以使用docker run
命令創建容器,如代碼清單3-2所示。docker run
命令提供了Docker容器的創建到啟動的功能,在本書中我們也會使用該命令來創建新容器。
代碼清單3-2 創建第一個容器
$ sudo docker run -i -t ubuntu /bin/bash Pulling repository ubuntu from https://index.docker.io/v1 Pulling image 8dbd9e392a964056420e5d58ca5cc376ef18e2de93b5cc90e868a1bbc8318c1c (precise) from ubuntu Pulling 8dbd9e392a964056420e5d58ca5cc376ef18e2de93b5cc90e868a1bbc8318c1c metadata Pulling 8dbd9e392a964056420e5d58ca5cc376ef18e2de93b5cc90e868a1bbc8318c1c fs layer Downloading 58337280/? (n/a) Pulling image b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc (quantal) from ubuntu Pulling image 27cf784147099545 () from ubuntu root@fcd78e1a3569:/#
{提示}
官方文檔列出了完整的Docker命令列表,你也可以使用docker help
獲取這些命令。此外,你還可以使用Docker的man
頁(即執行man docker-run
)。
代碼清單3-3所示命令的輸出結果非常豐富,下面我們來逐條解析。
代碼清單3-3 docker run
命令
$ sudo docker run -i -t ubuntu /bin/bash
首先,我們告訴Docker執行docker run
命令,并指定了-i
和-t
兩個命令行參數。-i
標志保證容器中STDIN
是開啟的,盡管我們并沒有附著到容器中。持久的標準輸入是交互式shell的“半邊天”,-t
標志則是另外“半邊天”,它告訴Docker為要創建的容器分配一個偽tty終端。這樣,新創建的容器才能提供一個交互式shell。若要在命令行下創建一個我們能與之進行交互的容器,而不是一個運行后臺服務的容器,則這兩個參數已經是最基本的參數了。
{提示}
官方文檔上列出了docker run
命令的所有標志,此外還可以用命令docker help run
查看這些標志。或者,也可以用Docker的man
頁(也就是執行man docker-run
命令)。
接下來,我們告訴Docker基于什么鏡像來創建容器,示例中使用的是ubuntu
鏡像。ubuntu
鏡像是一個常備鏡像,也可以稱為“基礎”(base)鏡像,它由Docker公司提供,保存在Docker HubRegistry上。
你可以用 ubuntu
基礎鏡像(以及類似的 fedora
、debian
、centos
等鏡像)為基礎,在你選擇的操作系統上構建自己的鏡像。這里,我們基于此基礎鏡像啟動了一個容器,并且沒有對容器進行任何改動。
那么,在這一切的背后又都發生了什么呢?首先Docker會檢查本地是否存在ubuntu
鏡像,如果本地還沒有該鏡像的話,那么Docker就會連接官方維護的Docker Hub Registry,查看Docker Hub中是否有該鏡像。Docker一旦找到該鏡像,就會下載該鏡像并將其保存到本地宿主機中。
隨后,Docker在文件系統內部用這個鏡像創建了一個新容器。該容器擁有自己的網絡、IP地址,以及一個用來和宿主機進行通信的橋接網絡接口。最后,我們告訴Docker在新容器中要運行什么命令,在本例中我們在容器中運行/bin/bash
命令啟動了一個Bash shell。
當容器創建完畢之后,Docker就會執行容器中的/bin/bash
命令,這時我們就可以看到容器內的shell了,就像代碼清單3-4所示。
代碼清單3-4 第一個容器的shell
root@f7cbdac22a02:/#
{注意}
在第4章中,我們將會看到如何構建自己的鏡像并基于該鏡像創建容器的基礎知識。
3 使用第一個容器
現在,我們已經以root
用戶登錄到了新容器中,容器的ID f7cbdac22a02``,乍``看起來有些令人迷惑的字符串
。這是一個完整的Ubuntu系統,你可以用它來做任何事情。下面我們就來研究一下這個容器。首先,我們可以獲取該容器的主機名,如代碼清單3-5所示。
代碼清單3-5 檢查容器的主機名
root@f7cbdac22a02:/# hostname f7cbdac22a02
可以看到,容器的主機名就是該容器的ID。我們再來看看/etc/hosts
文件,如代碼清單3-6所示。
代碼清單3-6 檢查容器的/etc/hosts文件
root@f7cbdac22a02:/# cat /etc/hosts 172.17.0.4 f7cbdac22a02 127.0.0.1 localhost ::1 localhost ip6-localhost ip6-loopback fe00::0 ip6-localnet ff00::0 ip6-mcastprefix ff02::1 ip6-allnodes ff02::2 ip6-allrouters
Docker以在hosts
文件中為該容器的IP地址添加了一條主機配置項。我們再來看看容器的網絡配置情況,如代碼清單3-7所示。
代碼清單3-7 檢查容器的接口
root@f7cbdac22a02:/# ip a 1: lo: <LOOPBACK,UP,LOWER_UP> mtu 1500 qdisc noqueue state UNKNOWN group default link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00 inet 127.0.0.1/8 scope host lo inet6 ::1/128 scope host valid_lft forever preferred_lft forever 899: eth0: <BROADCAST,UP,LOWER_UP> mtu 1500 qdisc pfifo_fast state UP group default qlen 1000 link/ether 16:50:3a:b6:f2:cc brd ff:ff:ff:ff:ff:ff inet 172.17.0.4/16 scope global eth0 inet6 fe80::1450:3aff:feb6:f2cc/64 scope link valid_lft forever preferred_lft forever
我們可以看到,這里有lo
的環回接口,還有IP為172.17.0.4
的標準eth0
網絡接口,和普通宿主機是完全一樣的。我們還可以查看容器中運行的進程,如代碼清單3-8所示。
代碼清單3-8 檢查容器的進程
root@f7cbdac22a02:/# ps -aux USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND root 1 0.0 0.0 18156 1936 ? Ss May30 0:00 /bin/bash root 21 0.0 0.0 15568 1100 ? R+ 02:38 0:00 ps -aux
接下來我們要干些什么呢?安裝一個軟件包怎么樣?如代碼清單3-9所示。
代碼清單3-9 在第一個容器中安裝軟件包
root@f7cbdac22a02:/# apt-get update && apt-get install vim
通過上述命令,我們就在容器中安裝了Vim軟件。
你可以繼續在容器中做任何自己想做的事情。當所有工作都結束時,輸入exit
,就可以返回到Ubuntu宿主機的命令行提示符了。
這個容器現在怎樣了?容器現在已經停止運行了!只有在指定的/bin/bash
命令處于運行狀態的時候,我們容器也才會相應地處于運行狀態。一旦退出容器,/bin/bash
命令也就結束了,這時容器也隨之停止了運行。
但容器仍然是存在的,我們可以用docker ps -a
命令查看當前系統中容器的列表
默認情況下,當執行docker ps
命令時,只能看到正在運行的容器。如果指定-a
標志,選項的話,那么docker ps
命令會列出所有容器,包括正在運行的和已經停止的。
{提示}
你也可以為docker ps
命令指定-l
標志,該選項會列出最后一次運行的容器,包括正在運行和已經停止的。
從該命令的輸出結果中我們可以看到關于這個容器的很多有用信息:ID、用于創建該容器的鏡像、容器最后執行的命令、創建時間以及容器的退出狀態(在上面的例子中,退出狀態是0
,因為容器是通過正常的exit
命令退出的)。我們還可以看到,每個容器都有一個名稱。
{注意}
有三種方式可以指代唯一容器:短UUID(如f7cbdac22a02
)、長UUID(如f7cbdac22a02e03c9438c729345e54db9d20cfa2ac1fc3494b6eb60872e74778
)或者名稱(如gray_cat
)。
4 容器命名
Docker會為我們創建的每一個容器自動生成一個隨機的名稱。例如,上面我們剛剛創建的容器就被命名為gray_cat
。如果想為容器指定一個名稱,而不是使用自動生成的名稱,則可以用--name
標志來實現,如代碼清單3-10所示。
代碼清單3-10 給容器命名
$ sudo docker run --name bob_the_container -i -t ubuntu /bin/bash root@aa3f365f0f4e:/# exit
上述命令將會創建一個名為bob_the_container
的容器。一個合法的容器名稱只能包含以下字符:小寫字母a~z、大寫字母A~Z、數字0~9、下劃線、圓點、橫線(如果用正則表達式來表示這些符號,就是[a-zA-Z0-9_.-]
)。
在很多Docker命令中,我們都可以用容器的名稱來替代容器ID,后面我們將會看到。容器名稱有助于分辨容器,當構建容器和應用程序之間的邏輯連接時,容器的名稱也有助于從邏輯上理解連接關系。具體的名稱(如web
、db
)比容器ID和隨機容器名好記多了。我推薦大家都使用容器名稱,以更加方便地管理容器。
容器的命名必須是唯一的。如果我們試圖創建兩個名稱相同的容器,則命令將會失敗。如果要使用的容器名稱已經存在,可以先用docker rm
命令刪除已有的同名容器后,再來創建新的容器。
5 重新啟動已經停止的容器
bob_the_container
容器已經停止了,接下來我們能對它做些什么呢?如果愿意,我們可以用下面的命令重新啟動一個已經停止的容器,如代碼清單3-11所示。
代碼清單3-11 啟動已經停止運行的容器
$ sudo docker start bob_the_container
除了容器名稱,我們也可以用容器ID來指定容器,如代碼清單3-12所示。
代碼清單3-12 通過ID啟動已經停止運行的容器
$ sudo docker start aa3f365f0f4e
{提示}
我們也可以使用docker restart
命令來重新啟動一個容器。
這時運行不帶-a
標志的docker ps
命令,就應該看到我們的容器已經開始運行了。
6 附著到容器上
Docker容器重新啟動的時候,會沿用docker run
命令時指定的參數來運行,因此我們容器重新啟動后會運行一個交互式會話shell。此外,我們也可以用docker attach
命令,重新附著到該容器的會話上,如代碼清單3-13所示。
代碼清單3-13 附著到正在運行的容器
$ sudo docker attach bob_the_container
我們也可以使用容器ID,重新附著到容器的會話上,如代碼清單3-14所示。
代碼清單3-14 通過ID附著到正在運行的容器
$ sudo docker attach aa3f365f0f4e
現在,我們又重新回到了容器的Bash提示符,如代碼清單3-15所示。
代碼清單3-15 重新附著到容器的會話
root@aa3f365f0f4e:/#
{提示}
你可能需要按下回車
鍵才能進入該會話。
如果退出容器的shell,容器也會隨之停止運行。
7 創建守護式容器
除了這些交互式運行的容器(interactive container),我們也可以創建長期運行的容器。守護式容器(daemonized container)沒有交互式會話,非常適合運行應用程序和服務。大多數時候我們都需要以守護式來運行我們的容器。下面我們就來啟動一個守護式容器,如 代碼清單3-16所示。
代碼清單3-16 創建長期運行的容器
$ sudo docker run --name daemon_dave -d ubuntu /bin/sh -c "while true; do echo hello world; sleep 1; done" 1333bb1a66af402138485fe44a335b382c09a887aa9f95cb9725e309ce5b7db3
我們在上面的docker run
命令使用了-d
參數,因此Docker會將容器放到后臺運行。
我們還在容器要運行的命令里使用了一個while
循環,該循環會一直打印hello world
,直到容器或其進程停止運行。
通過組合使用上面的這些參數,你可以發現docker run
命令并沒有像上一個容器一樣將主機的控制臺附著到新的shell會話上,而是僅僅返回了一個容器ID而已,我們還是在主機的命令行之中。如果我們執行docker ps
命令,可以看到一個正在運行的容器,如代碼清單3-17所示。
代碼清單3-17 查看正在運行的daemon_dave
容器
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES 1333bb1a66af ubuntu:14.04 /bin/sh -c 'while tr 32 secs ago Up 27 daemon_dave
8容器內部都在干些什么
現在我們已經有了一個在后臺運行while
循環的守護型容器。為了探究該容器內部都在干些什么,我們可以用docker logs
命令來獲取容器的日志,如代碼清單3-18所示。
代碼清單3-18 獲取守護式容器的日志
$ sudo docker logs daemon_dave hello world hello world hello world hello world hello world hello world hello world . . .
這里,我們可以看到while
循環正在向日志里打印hello world
。Docker會輸出最后幾條日志項并返回。我們也可以在命令后使用-f
參數來監控Docker的日志,這與tail -f
命令非常相似,如代碼清單3-19所示。
代碼清單3-19 跟蹤守護式容器的日志
$ sudo docker logs -f daemon_dave hello world hello world hello world hello world hello world hello world hello world . . .
{提示}
可以通過Ctrl+C
退出日志跟蹤。
我們也可以跟蹤容器日志的某一片段,和之前類似,只需要在tail
命令后加入-f --lines
標志即可。例如,可以用docker logs --tail 10 daemon_dave
獲取日志的最后10行內容。另外,也可以用docker logs --tail 0 -f daemon_dave
命令來跟蹤某個容器的最新日志而不必讀取整個日志文件。
為了讓調試更簡單,我們還可以使用-t
標志為每條日志項加上時間戳,如代碼清單3-20所示。
代碼清單3-20 跟蹤守護式容器的最新日志
$ sudo docker logs -ft daemon_dave [May 10 13:06:17.934] hello world [May 10 13:06:18.935] hello world [May 10 13:06:19.937] hello world [May 10 13:06:20.939] hello world [May 10 13:06:21.942] hello world . . .
{提示}
同樣,可以通過Ctr+C
退出日志跟蹤。
9 查看容器內的進程
除了容器的日志,我們也可以查看容器內部運行的進程。要做到這一點,要使用docker top
命令,如代碼清單3-21所示。
代碼清單3-21 查看守護式容器的進程
$ sudo docker top daemon_dave
該命令執行后,我們可以看到容器內的所有進程(主要還是我們的while
循環)、運行進程的用戶及進程ID,如代碼清單3-22所示。
代碼清單3-22 docker``top
命令的輸出結果
PID USER COMMAND 977 root /bin/sh -c while true; do echo hello world; sleep 1; done 1123 root sleep 1
10 在容器內部運行進程
在Docker 1.3之后,我們也可以通過docker exec
命令在容器內部額外啟動新進程。可以在容器內運行的
進程有兩種類型:后臺任務和交互式任務。后臺任務在容器內運行且沒有交互需求,而交互式任務則保持在前臺運行。對于需要在容器內部打開shell的任務,
交互式任務是很實用的。下面我們先來看一個后臺任務的例子,如代碼清單3-23所示。
代碼清單3-23 在容器中運行后臺任務
$ sudo docker exec -d daemon_dave touch /etc/new_config_file
這里的-d
標志表明需要運行一個后臺進程,-d
標志之后,指定的是要在內部執行這個命令的容器的名字以及要執行的命令。上面例子中的命令會在daemon_dave
容器內創建了一個空文件,文件名為/etc/new_config_file
。通過docker exec
后臺命令,我們可以在正在運行的容器中進行維護、監控及管理任務。
我們也可以在daemon_dave
容器中啟動一個諸如打開shell的交互式任務,如代碼清單3-24所示。
代碼清單3-24 在容器內運行交互命令
$ sudo docker exec -t -i daemon_dave /bin/bashVersion:
和運行交互容器時一樣,這里的-t
和-i
標志為我們執行的進程創建了TTY并捕捉STDIN
。接著我們指定了要在內部執行這個命令的容器的名字以及要執行的命令。在上面的例子中,這條命令會在daemon_dave
容器內創建一個新的bash會話,有了這個會話,我們就可以在該容器中運行其他命令了。
{注意}
docker exec
命令是Docker 1.3引入的,早期版本并不支持該命令。對于早期Docker版本,請參考第6章中介紹的nsenter
命令。
11 停止守護式容器
要停止守護式容器,只需要執行docker stop
命令,如代碼清單3-25所示。
代碼清單3-25 停止正在運行的Docker容器
$ sudo docker stop daemon_dave
當然,也可以用容器ID來指代容器名稱,如代碼清單3-26所示。
代碼清單3-26 通過容器ID停止正在運行的容器
$ sudo docker stop c2c4e57c12c4
{注意}
docker stop
命令會向Docker容器進程發送SIGTERM
信號。如果你想快速停止某個容器,也可以使用docker kill
命令來向容器進程發送SIGKILL
信號。
要想查看已經停止的容器的狀態,則可以使用docker ps
命令。還有一個很實用的命令docker ps -n x
,該命令會顯示最后x
個容器,不論這些容器正在運行還是已經停止。
12 自動重啟容器
如果由于某種錯誤而導致容器停止運行,我們還可以通過--restart
標志,讓Docker自動重新啟動該容器。--restart
標志會檢查容器的退出代碼,并據此來決定是否要重啟容器。默認的行為是Docker不會重啟容器。
代碼清單3-27是一個在docker run
命令中使用—restart
標志的例子。
代碼清單3-27 自動重啟容器
$ sudo docker run --restart=always --name daemon_dave -d ubuntu / bin/sh -c "while true; do echo hello world; sleep 1; done"
在本例中,--restart
標志被設置為always
。無論容器的退出代碼是什么,Docker都會自動重啟該容器。除了always
,我們還可以將這個標志設為on-failure
,這樣,只有當容器的退出代碼為非0值的時候,才會自動重啟。另外,on-failure``還接受``一個可選的
重啟次數參數,如代碼清單3-28所示。
代碼清單3-28 為on-failure
指定count
參數
--restart=on-failure:5
這樣,當容器退出代碼為非0時,Docker會嘗試自動重啟該容器,最多重啟5次。
{注意}
--restart
標志是Docker1.2.0引入的選項。
13 深入容器
除了通過docker ps
命令獲取容器的信息,我們還可以使用docker inspect``來
獲得更多的容器信息,如代碼清單3-29所示。
代碼清單3-29 查看容器
$ sudo docker inspect daemon_dave [{ "ID": "c2c4e57c12c4c142271c031333823af95d64b20b5d607970c334784430bcbd0f", "Created": "2014-05-10T11:49:01.902029966Z", "Path": "/bin/sh", "Args": [ "-c", "while true; do echo hello world; sleep 1; done" ], "Config": { "Hostname": "c2c4e57c12c4", . . .
docker inspect
命令會對容器進行詳細的檢查,然后返回其配置信息,包括名稱、命令、網絡配置以及很多有用的數據。
我們也可以用-f
或者--format
標志來選定查看結果,如代碼清單3-30所示。
代碼清單3-30 有選擇地獲取容器信息
$ sudo docker inspect --format='{{ .State.Running }}' daemon_dave false
上面這條命令會返回容器的運行狀態,示例中該狀態為false
。我們還能獲取其他有用的信息,如容器IP地址,如代碼清單3-31所示。
代碼清單3-31 查看容器的IP地址
$ sudo docker inspect --format '{{ .NetworkSettings.IPAddress }}' \ daemon_dave 172.17.0.2
{提示}
--format
或者-f
標志遠非表面看上去那么簡單。該標志實際上支持完整的Go語言模板。用它進行查詢時,可以充分利用Go語言模板的優勢。
我們也可以同時指定多個容器,并顯示每個容器的輸出結果,如代碼清單3-32所示。
代碼清單3-32 查看多個容器
$ sudo docker inspect --format '{{.Name}} {{.State.Running}}' \ daemon_dave bob_the_container /daemon_dave false /bob_the_container false
我們可以為該參數指定要查詢和返回的查看散列(inspect hash)中的任意部分。
{注意}
除了查看容器,你還可以通過瀏覽/var/lib/docker
目錄來深入了解Docker的工作原理。該目錄存放著Docker鏡像、容器以及容器的配置。所有的容器都保存在/var/lib/docker/containers
目錄下。
14 刪除容器
如果容器已經不再使用,可以使用docker rm
命令來刪除它們,如代碼清單3-33所示。
代碼清單3-33 刪除容器
$ sudo docker rm 80430f8d0921 80430f8d0921
{注意}
需要注意的是,運行中的Docker容器是無法刪除的!你必須先通過docker stop
或docker kill
命令停止容器,才能將其刪除。
目前,還沒有辦法一次刪除所有容器,不過可以通過代碼清單3-34所示的小技巧來刪除全部容器。
代碼清單3-34 刪除所有容器
docker rm `docker ps -a -q`
上面的docker ps
命令會列出現有的全部容器,-a
標志代表列出所有(all)容器,而-q
標志則表示只需要返回容器的ID而不會返回容器的其他信息。這樣我們就得到了容器ID的列表,并傳給了docker rm
命令,從而達到刪除所有容器的目的。
小結
在本章中我們介紹了Docker容器的基本工作原理。這里學到的內容也是本書剩余章節中學習如何使用Docker的基礎。
本文摘自《第一本Docker書》

- Docker公司前服務與支持副總裁力作!
- 全球第一本Docker技術圖書!
- 學習Docker必備的第一本書!
來自:http://www.jianshu.com/p/26f15063de7d