欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Dockerfile中CMD和ENTRYPOINT命令詳解

 更新時間:2018年03月12日 11:46:00   作者:sparkdev  
Dockerfile中的ENTRYPOINT指令和CMD指令都可以設置容器啟動時要執(zhí)行的命令,但用途是有略微不同的。下面這篇文章主要給大家介紹了關(guān)于Dockerfile中CMD和ENTRYPOINT命令的相關(guān)資料,需要的朋友可以參考借鑒,下面隨著小編來一起學習學習吧。

前言

CMD 和 ENTRYPOINT 指令都是用來指定容器啟動時運行的命令。

單從功能上來看,這兩個命令幾乎是重復的。單獨使用其中的一個就可以實現(xiàn)絕大多數(shù)的用例。但是既然 doker 同時提供了它們,為了在使用中不至于混淆,本文試圖把它們的用法理清楚。下面話不多說了,來一起看看詳細的介紹吧。

exec 模式和 shell 模式

CMD 和 ENTRYPOINT 指令都支持 exec 模式和 shell 模式的寫法,所以要理解 CMD 和 ENTRYPOINT 指令的用法,就得先區(qū)分 exec 模式和 shell 模式。這兩種模式主要用來指定容器中的不同進程為 1 號進程。了解 linux 的朋友應該清楚 1 號進程在系統(tǒng)中的重要地位。筆者也在《在 docker 容器中捕獲信號》一文中介紹過 1 號進程對容器中信號處理的重要性,感興趣的朋友可以移步這里進行了解。下面我們通過 CMD 指令來學習 exec 模式和 shell 模式的特點。

exec 模式

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

FROM ubuntu
CMD [ "top" ]

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

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

然后查看容器中的進程 ID:

$ docker exec testcon ps aux

從圖中我們看到運行 top 命令的進程 ID 為 1。

exec 模式是建議的使用模式,因為當運行任務的進程作為容器中的 1 號進程時,我們可以通過 docker 的 stop 命令優(yōu)雅的結(jié)束容器(詳情請參考《在 docker 容器中捕獲信號》)。

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

FROM ubuntu
CMD [ "echo", "$HOME" ]

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

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

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

FROM ubuntu
CMD [ "sh", "-c", "echo $HOME" ]

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

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

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

shell 模式

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

FROM ubuntu
CMD top

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

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

然后查看容器中的進程 ID:

$ docker exec testcon2 ps aux

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

CMD 指令

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

CMD 指令有三種使用方式,其中的一種是為 ENTRYPOINT 提供默認的參數(shù):

CMD ["param1","param2"]

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

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

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

FROM ubuntu
CMD [ "top" ]

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

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

ENTRYPOINT 指令

ENTRYPOINT 指令的目的也是為容器指定默認執(zhí)行的任務。

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

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

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

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

FROM ubuntu
ENTRYPOINT [ "top", "-b" ]

運行下面的命令:

$ docker run --rm test1 -c

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

由 CMD 指令指定默認的可選參數(shù):

FROM ubuntu
ENTRYPOINT [ "top", "-b" ]
CMD [ "-c" ]

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

$ docker run --rm test2

這時容器中運行的命令為:top -b -c。

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

$ docker run --rm test2 -n 1

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

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

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

FROM ubuntu
ENTRYPOINT echo $HOME 

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

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

覆蓋默認的 ENTRYPOINT 指令:

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

$ docker run --rm --entrypoint hostname test2

這里我們使用 hostname 命令覆蓋了默認的 echo $HOME 命令。

Dockerfile 中至少要有一個

如果鏡像中既沒有指定 CMD 也沒有指定 ENTRYPOINT 那么在啟動容器時會報錯。這不算是什么問題,因為現(xiàn)在能見到的絕大多數(shù)鏡像都默認添加了 CMD 或 ENTRYPOINT 指令。

指定任意一個,效果差不多

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

或者:

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

同時使用 CMD 和 ENTRYPOINT 的情況

對于 CMD 和 ENTRYPOINT 的設計而言,多數(shù)情況下它們應該是單獨使用的。當然,有一個例外是 CMD 為 ENTRYPOINT 提供默認的可選參數(shù)。

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

     • 如果 ENTRYPOINT 使用了 shell 模式,CMD 指令會被忽略。

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

     • 如果 ENTRYPOINT 使用了 exec 模式,CMD 也應該使用 exec 模式。

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

當我們無法理解容器中運行命令的行為時,說不定通過這個表格可以解開疑惑!

總結(jié)

對于 Dockerfile 來說,CMD 和 ENTRYPOINT 是非常重要的指令。它們不是在構(gòu)建鏡像的過程中執(zhí)行,而是在啟動容器時執(zhí)行,所以主要用來指定容器默認執(zhí)行的命令。但是提供兩個功能類似的指令,必然會給用戶帶來理解上的困惑和使用中的混淆。希望本文能夠幫助大家理解二者的區(qū)別與聯(lián)系,并更好的使用二者。

參考:

Docker 官方文檔
ENTRYPOINT vs CMD: Back to Basics
Dockerfile: ENTRYPOINT vs CMD

相關(guān)文章

  • 解決Docker啟動Elasticsearch7.x報錯的問題

    解決Docker啟動Elasticsearch7.x報錯的問題

    這篇文章主要介紹了解決Docker啟動Elasticsearch7.x報錯的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • Docker容器日志查看和清理的圖文教程

    Docker容器日志查看和清理的圖文教程

    docker容器在運行過程中會產(chǎn)生日志,日志的量跟具體的容器有關(guān),下面這篇文章主要給大家介紹了關(guān)于Docker容器日志查看和清理的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-02-02
  • 如何設置docker開機自啟動,并設置容器自動重啟

    如何設置docker開機自啟動,并設置容器自動重啟

    這篇文章主要介紹了如何設置docker開機自啟動,并設置容器自動重啟問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • docker安裝elasticsearch和kibana的方法步驟

    docker安裝elasticsearch和kibana的方法步驟

    這篇文章主要介紹了docker安裝elasticsearch和kibana的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-06-06
  • Docker安裝OpenWrt的實現(xiàn)步驟

    Docker安裝OpenWrt的實現(xiàn)步驟

    OpenWrt是一種可以運行在路由器上的嵌入式操作系統(tǒng),提供了許多網(wǎng)絡服務和功能,本文主要介紹了Docker安裝OpenWrt的實現(xiàn)步驟,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-08-08
  • 利用Docker分層構(gòu)建優(yōu)化鏡像大小的實現(xiàn)

    利用Docker分層構(gòu)建優(yōu)化鏡像大小的實現(xiàn)

    合適docker鏡像文件大小不僅影響容器啟動效率,也影響資源占用效率,本文介紹如何利用分層方式構(gòu)建docker鏡像,采用多種方式避免鏡像文件太大而影響性能,需要的朋友可以參考下
    2025-01-01
  • docker啟動時環(huán)境變量不生效的解決方法

    docker啟動時環(huán)境變量不生效的解決方法

    因項目需要多處部署,所以打包成docker鏡像以便于部署,本文主要介紹了docker啟動時環(huán)境變量不生效的解決方法,具有一定的參考價值,感興趣的可以了解一下
    2024-03-03
  • Docker?配置容器固定IP的方法

    Docker?配置容器固定IP的方法

    這篇文章主要介紹了Docker?配置容器固定IP,下面使用docker自帶的network實現(xiàn)固定ip分配,并且重啟不會消失,通過綁定步驟給大家介紹的非常詳細,需要的朋友參考下吧
    2022-04-04
  • Docker實現(xiàn)掛載的方式匯總

    Docker實現(xiàn)掛載的方式匯總

    Docker實現(xiàn)掛載有多種方式,下面給大家分享Docker實現(xiàn)掛載常見方式,感興趣的朋友跟隨小編一起看看吧
    2023-11-11
  • Docker desktop安裝mysql過程

    Docker desktop安裝mysql過程

    文章介紹了如何在本地已有的Docker環(huán)境中拉取并啟動MySQL容器,并詳細說明了常用的環(huán)境變量參數(shù),此外,還討論了如何通過掛接本地磁盤來持久化MySQL數(shù)據(jù),從而避免數(shù)據(jù)丟失
    2024-12-12

最新評論