溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Dockerfile中的CMD和ENTRYPOINT命令怎么用

發(fā)布時(shí)間:2022-05-20 15:21:18 來源:億速云 閱讀:642 作者:iii 欄目:大數(shù)據(jù)

這篇文章主要介紹了Dockerfile中的CMD和ENTRYPOINT命令怎么用的相關(guān)知識,內(nèi)容詳細(xì)易懂,操作簡單快捷,具有一定借鑒價(jià)值,相信大家閱讀完這篇Dockerfile中的CMD和ENTRYPOINT命令怎么用文章都會有所收獲,下面我們一起來看看吧。

exec 模式

使用 exec 模式時(shí),容器中的任務(wù)進(jìn)程就是容器內(nèi)的 1 號進(jìn)程,看下面的例子:

from ubuntu
cmd [ "top" ]

把上面的代碼保存到 test1 目錄的 dockerfile 中,然后進(jìn)入 test1 目錄構(gòu)建鏡像并啟動(dòng)一個(gè)容器:

$ docker build -t test1 .
$ docker run -idt --name testcon test1

然后查看容器中的進(jìn)程 id:

$ docker exec testcon ps aux

Dockerfile中的CMD和ENTRYPOINT命令怎么用

從圖中我們看到運(yùn)行 top 命令的進(jìn)程 id 為 1。

exec 模式是建議的使用模式,因?yàn)楫?dāng)運(yùn)行任務(wù)的進(jìn)程作為容器中的 1 號進(jìn)程時(shí),我們可以通過 docker 的 stop 命令優(yōu)雅的結(jié)束容器(詳情請參考《》)。

exec 模式的特點(diǎn)是不會通過 shell 執(zhí)行相關(guān)的命令,所以像 $home 這樣的環(huán)境變量是取不到的:

from ubuntu
cmd [ "echo", "$home" ]

把上面的代碼保存到 test1 目錄的 dockerfile 中,然后進(jìn)入 test1 目錄構(gòu)建鏡像并啟動(dòng)一個(gè)容器:

$ docker build --no-cache -t test1 .
$ docker run --rm test1

Dockerfile中的CMD和ENTRYPOINT命令怎么用

通過 exec 模式執(zhí)行 shell 可以獲得環(huán)境變量:

from ubuntu
cmd [ "sh", "-c", "echo $home" ]

把上面的代碼保存到 test1 目錄的 dockerfile 中,然后進(jìn)入 test1 目錄構(gòu)建鏡像并啟動(dòng)一個(gè)容器:

$ docker build --no-cache -t test1 .
$ docker run --rm test1

Dockerfile中的CMD和ENTRYPOINT命令怎么用

這次正確取到了 $home 環(huán)境變量的值。

shell 模式

使用 shell 模式時(shí),docker 會以 /bin/sh -c "task command" 的方式執(zhí)行任務(wù)命令。也就是說容器中的 1 號進(jìn)程不是任務(wù)進(jìn)程而是 bash 進(jìn)程,看下面的例子:

from ubuntu
cmd top

把上面的代碼保存到 test2 目錄的 dockerfile 中,然后進(jìn)入 test2 目錄構(gòu)建鏡像并啟動(dòng)一個(gè)容器:

$ docker build -t test2 .
$ docker run -itd --name testcon2 test2

然后查看容器中的進(jìn)程 id:

$ docker exec testcon2 ps aux

Dockerfile中的CMD和ENTRYPOINT命令怎么用

1 號進(jìn)程執(zhí)行的命令居然是 /bin/sh -c top。而我們指定的 top 命令的進(jìn)程 id 為 7。這是由 docker 內(nèi)部決定的,目的是讓我們執(zhí)行的命令或者腳本可以取到環(huán)境變量。

cmd 指令

cmd 指令的目的是:為容器提供默認(rèn)的執(zhí)行命令。

cmd 指令有三種使用方式,其中的一種是為 entrypoint 提供默認(rèn)的參數(shù):

cmd ["param1","param2"]

另外兩種使用方式分別是 exec 模式和 shell 模式:

cmd ["executable","param1","param2"] // 這是 exec 模式的寫法,注意需要使用雙引號。
cmd command param1 param2 // 這是 shell 模式的寫法。

注意命令行參數(shù)可以覆蓋 cmd 指令的設(shè)置,但是只能是重寫,卻不能給 cmd 中的命令通過命令行傳遞參數(shù)。
一般的鏡像都會提供容器啟動(dòng)時(shí)的默認(rèn)命令,但是有些場景中用戶并不想執(zhí)行默認(rèn)的命令。用戶可以通過命令行參數(shù)的方式覆蓋 cmd 指令提供的默認(rèn)命令。比如通過下面命令創(chuàng)建的鏡像:

from ubuntu
cmd [ "top" ]

在啟動(dòng)容器時(shí)我們通過命令行指定參數(shù) ps aux 覆蓋默認(rèn)的 top 命令:

Dockerfile中的CMD和ENTRYPOINT命令怎么用

從上圖可以看到,命令行上指定的 ps aux 命令覆蓋了 dockerfile 中的 cmd [ "top" ]。實(shí)際上,命令行上的命令同樣會覆蓋 shell 模式的 cmd 指令。

entrypoint 指令

entrypoint 指令的目的也是為容器指定默認(rèn)執(zhí)行的任務(wù)。

entrypoint 指令有兩種使用方式,就是我們前面介紹的 exec 模式和 shell 模式:

entrypoint ["executable", "param1", "param2"] // 這是 exec 模式的寫法,注意需要使用雙引號。
entrypoint command param1 param2 // 這是 shell 模式的寫法。

exec 模式和 shell 模式的基本用法和 cmd 指令是一樣的,下面我們介紹一些比較特殊的用法。

指定 entrypoint 指令為 exec 模式時(shí),命令行上指定的參數(shù)會作為參數(shù)添加到 entrypoint 指定命令的參數(shù)列表中。用下面的代碼構(gòu)建鏡像 test1:

from ubuntu
entrypoint [ "top", "-b" ]

運(yùn)行下面的命令:

$ docker run --rm test1 -c

Dockerfile中的CMD和ENTRYPOINT命令怎么用

我們在命令行上添加的參數(shù)被追加到了 top 命令的參數(shù)列表中。

由 cmd 指令指定默認(rèn)的可選參數(shù):

from ubuntu
entrypoint [ "top", "-b" ]
cmd [ "-c" ]

使用這段代碼構(gòu)建鏡像 test2 并不帶命令行參數(shù)啟動(dòng)容器:

$ docker run --rm test2

這時(shí)容器中運(yùn)行的命令為:top -b -c。

如果我們指定命令行參數(shù):

$ docker run --rm test2 -n 1

-n 1 會覆蓋 通過 cmd [ "-c" ] 指定的參數(shù),容器執(zhí)行的命令為:top -b -n 1

Dockerfile中的CMD和ENTRYPOINT命令怎么用

注意上圖的輸出顯示 -c 參數(shù)被覆蓋了。

指定 entrypoint 指令為 shell 模式時(shí),會完全忽略命令行參數(shù):

from ubuntu
entrypoint echo $home

把上面的代碼編譯成鏡像 test2,分別不帶命令行參數(shù)和使用命令行參數(shù) ls 執(zhí)行命令:

Dockerfile中的CMD和ENTRYPOINT命令怎么用

我們看到 ls 命令沒有被執(zhí)行,這說明命令行參數(shù)被 entrypoint 指令的 shell 模式忽略了。

覆蓋默認(rèn)的 entrypoint 指令:

entrypoint 指令也是可以被命令行覆蓋的,只不過不是默認(rèn)的命令行參數(shù),而是需要顯式的指定 --entrypoint 參數(shù)。比如我們通過下面的方式覆蓋上面鏡像中的 echo $home 命令:

$ docker run --rm --entrypoint hostname test2

Dockerfile中的CMD和ENTRYPOINT命令怎么用

這里我們使用 hostname 命令覆蓋了默認(rèn)的 echo $home 命令。

dockerfile 中至少要有一個(gè)

如果鏡像中既沒有指定 cmd 也沒有指定 entrypoint 那么在啟動(dòng)容器時(shí)會報(bào)錯(cuò)。這不算是什么問題,因?yàn)楝F(xiàn)在能見到的絕大多數(shù)鏡像都默認(rèn)添加了 cmd 或 entrypoint 指令。

指定任意一個(gè),效果差不多

從結(jié)果上看,cmd 和 entrypoint 是一樣的,我們可以通過它們實(shí)現(xiàn)相同的目的。下面我們分別用 cmd 和 entrypoint 設(shè)置 top -b 命令,然后觀察容器運(yùn)行時(shí)的 metadata 信息:

Dockerfile中的CMD和ENTRYPOINT命令怎么用

或者:

Dockerfile中的CMD和ENTRYPOINT命令怎么用

雖然實(shí)現(xiàn)方式不同,但最終容器運(yùn)行的命令是一樣的。

同時(shí)使用 cmd 和 entrypoint 的情況

對于 cmd 和 entrypoint 的設(shè)計(jì)而言,多數(shù)情況下它們應(yīng)該是單獨(dú)使用的。當(dāng)然,有一個(gè)例外是 cmd 為 entrypoint 提供默認(rèn)的可選參數(shù)。

我們大概可以總結(jié)出下面幾條規(guī)律:

     ? 如果 entrypoint 使用了 shell 模式,cmd 指令會被忽略。

     ? 如果 entrypoint 使用了 exec 模式,cmd 指定的內(nèi)容被追加為 entrypoint 指定命令的參數(shù)。

     ? 如果 entrypoint 使用了 exec 模式,cmd 也應(yīng)該使用 exec 模式。

真實(shí)的情況要遠(yuǎn)比這三條規(guī)律復(fù)雜,好在 docker 給出了官方的解釋,如下圖所示:

Dockerfile中的CMD和ENTRYPOINT命令怎么用

關(guān)于“Dockerfile中的CMD和ENTRYPOINT命令怎么用”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對“Dockerfile中的CMD和ENTRYPOINT命令怎么用”知識都有一定的了解,大家如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI