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

Dockerfile中ENTRYPOINT 和 CMD的區(qū)別說明

 更新時(shí)間:2020年11月21日 09:05:08   作者:Allocator  
這篇文章主要介紹了Dockerfile中ENTRYPOINT 和 CMD的區(qū)別說明,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧

在Docker的系統(tǒng)學(xué)習(xí)教程中我們了解到使用Dockerfile構(gòu)建Docker鏡像為一個(gè)規(guī)范的方式,根據(jù)Dockerfile可以了解鏡像中安裝的組件的詳細(xì)內(nèi)容。

Dockerfile一般由四部分組成:第一,構(gòu)建的基礎(chǔ)鏡像;第二,鏡像構(gòu)建者的信息;第三,構(gòu)建鏡像過程中鏡像層添加指令;第四,由該鏡像啟動(dòng)容器時(shí)執(zhí)行的程序。

本篇文章中涉及到的ENTRYPOINT 和CMD 屬于Dockerfile中的最后一部分,這兩個(gè)Dockerfile指令是用來告知Docker后臺(tái)程序啟動(dòng)鏡像時(shí)需要執(zhí)行的程序,兩者有細(xì)微的差別。

下面將從兩者的異同以及兩者聯(lián)合使用的高級(jí)技巧方面對兩個(gè)指令進(jìn)行詳解。

CMD 指令

CMD 指令指定容器啟動(dòng)時(shí)需要運(yùn)行的程序。一般用最簡單的方式啟動(dòng)一個(gè)容器時(shí)使用docker run 會(huì)傳遞參數(shù)給docker指令

docker run -it image /bin/bash

后面的/bin/bash 其實(shí)是傳遞參數(shù),告知容器啟動(dòng)時(shí)運(yùn)行一個(gè)shell。這個(gè)過程可以用CMD 指令等效的替換

CMD ['/bin/bash']

因此在Dockerfile中存在這個(gè)CMD指令指定的命令時(shí),啟動(dòng)容器就可以不進(jìn)行參數(shù)傳遞。

docker run -it image

執(zhí)行效果一致。

[root@MiWiFi-R3L-srv test]# docker run --name test -it test_image

[root@3a1bb0c9e35c /]#

如果dockerfile中已經(jīng)指定了容器啟動(dòng)時(shí)運(yùn)行的程序,同時(shí)在使用docker run 啟動(dòng)容器時(shí)使用了命令行參數(shù),那么dockerfile 中的cmd 指令將無效

docker run -it image /bin/ps

發(fā)現(xiàn)啟動(dòng)容器后沒有shell ,只是打印出了當(dāng)前容器中的進(jìn)程狀態(tài),cmd 指令效果被覆蓋。

 PID TTY     TIME CMD
  1 ?    00:00:00 ps
[root@MiWiFi-R3L-srv test]#

此時(shí)可以看到cmd 效果被覆蓋。在一個(gè)dockerfile中只有最后一個(gè)cmd 指令有效,因此一個(gè)dockerfile中只寫一個(gè)cmd 指令。

ENTRYPOINT 指令

ENTRYPOINT 指令效果與CMD 非常的類似,比較容易混淆兩者的功能。最大的區(qū)別在于使用的方式,ENTRYPOINT 指定的命令需要與docker run 啟動(dòng)容器進(jìn)行搭配,將docker run 指令后面跟的內(nèi)容當(dāng)做參數(shù)作為ENTRYPOINT指令指定的運(yùn)行命令的參數(shù),ENTRYPOINT 指定的linux命令一般是不會(huì)被覆蓋的。

以nginx 鏡像為例說明

首先構(gòu)建一個(gè)nginx鏡像,并且指定容器運(yùn)行時(shí)執(zhí)行的程序?yàn)閚ginx。

FROM centos
MAINTAINER allocator
RUN yum install -y nginx
RUN echo 'hello world' > /usr/share/nginx/html/index.html
EXPOSE 80
ENTRYPOINT ["/usr/sbin/nginx"]

然后啟動(dòng)鏡像

docker run --name test -p 5000:80 -it test_nginx -g "daemon off"

后面兩個(gè)是作為參數(shù)傳遞給nginx啟動(dòng)程序運(yùn)行,此時(shí)nginx作為前臺(tái)程序運(yùn)行,是一個(gè)web服務(wù)器,可以根據(jù)外部綁定的端口,通過瀏覽器正??吹絟ello world

兩者聯(lián)合使用技巧

已經(jīng)明白了兩者的區(qū)別,可以利用兩者的特點(diǎn)構(gòu)建一個(gè)含有默認(rèn)啟動(dòng)運(yùn)行程序的鏡像,并且支持docker run 啟動(dòng)時(shí)人為指定啟動(dòng)程序運(yùn)行的參數(shù)。

舉個(gè)例子。利用ENTRYPOINT 指定啟動(dòng)時(shí)運(yùn)行啟動(dòng)nginx程序,并給定默認(rèn)的運(yùn)行參數(shù)為顯示幫助信息,dockerfile 構(gòu)建如下:

ENTRYPOINT ["/usr/sbin/nginx"]

CMD ["-h"]

當(dāng)使用docker run --name test -it test_nginx 不傳遞任何參數(shù)時(shí),此時(shí)啟動(dòng)容器會(huì)使用cmd 指令后的命令作為默認(rèn)參數(shù),打印nginx的幫助信息。此時(shí)cmd 后的內(nèi)容并不是一個(gè)完整的指令,而是參數(shù),如果其內(nèi)容是一個(gè)完整的指令,那么它將覆蓋掉ENTRYPOINT 中的內(nèi)容。

如果使用docker run --name test -it test_nginx -g "daemon off" 啟動(dòng)時(shí),此時(shí)給定的運(yùn)行參數(shù)會(huì)覆蓋掉CMD 指令對應(yīng)的內(nèi)容,此時(shí)nginx將作為前臺(tái)進(jìn)程運(yùn)行,作為一個(gè)web服務(wù)器使用,通過browser可以看到hello world

補(bǔ)充知識(shí):docker-entrypoint.sh 入口文件編寫技巧

在 docker 的官方 Registry 中(store.docker.com) 流行的第三方應(yīng)用在自己的頁面中都提供了 dockerfile 的鏈接. 而很多 dockerfile 的 ENTRYPOINT 命令都是這么寫的 ["docker-entrypoint.sh"]

本篇文章就掃盲下 docker-entrypoint.sh 的特殊用法和設(shè)計(jì)邏輯

MySQL

set -e

你寫的每個(gè)腳本都應(yīng)該在文件開頭加上set -e, 這句語句告訴bash如果任何語句的執(zhí)行結(jié)果不是true則應(yīng)該退出. 這樣的好處是防止錯(cuò)誤像滾雪球般變大導(dǎo)致一個(gè)致命的錯(cuò)誤, 而這些錯(cuò)誤本應(yīng)該在之前就被處理掉. 如果要增加可讀性, 可以使用set -o errexit, 它的作用與set -e相同

set -o pipefail

設(shè)計(jì)用途同上, 就是希望在執(zhí)行錯(cuò)誤之后立即退出, 不要再向下執(zhí)行了. 而 -o pipefail 的作用域是管道, 也就是說在 Linux 腳本中的管道, 如果前面的命令執(zhí)行出了問題, 應(yīng)該立即退出

shopt -s nullglob

在使用 Linux 中的通配符時(shí) * ?等 如果沒有匹配到任何文件, 不會(huì)報(bào) No such file or directory 而是將命令后面的參數(shù)去掉執(zhí)行

if [ “${1:0:1}” = ‘-‘ ]; then…

這是一個(gè)判斷語句, 在官方文件中, 上一行已經(jīng)給出了注釋: if command starts with an option, prepend mysqld

這個(gè)判斷語句是 ${1:0:1} 意思是判斷 $1(調(diào)用該腳本的第一個(gè)參數(shù)), 偏移量0(不偏移), 取一個(gè)字符(取字符串的長度)

如果判斷出來調(diào)用這個(gè)腳本后面所跟的參數(shù)第一個(gè)字符是-中橫線的話, 就認(rèn)為后面的所有字符串都是 mysqld 的啟動(dòng)參數(shù)

上面的這個(gè)操作類似于 Python 的字符串切片

set – mysqld “$@”

在上面判斷完第一個(gè)參數(shù)是-開頭之后, 緊接著就執(zhí)行了 set -- mysqld "$@" 這個(gè)命令. 使用了 set -- 的用法. set —會(huì)將他后面所有以空格區(qū)分的字符串, 按順序分別存儲(chǔ)到$1, $2, $3 變量中, 其中新的$@ 為 set — 后面的全部內(nèi)容

舉例來說: bash docker-entrypoint.sh -f xxx.conf

在這種情況下, set -- mysqld "$@" 中的 $@ 的值為 -f xxx.conf

當(dāng)執(zhí)行完 set -- mysqld "$@" 這條命令后:

$1=mysqld
$2=-f
$3=xxx.conf
$@=mysqld -f xxx.conf

可以看到, 當(dāng)執(zhí)行 docker-entrypoint.sh腳本的時(shí)候后面加了 -x形式的參數(shù)之后, $@的值發(fā)生的改變, 在原有$@值的基礎(chǔ)之上, 在前面又預(yù)添加了 mysqld 命令

exec “$@”

幾乎在每個(gè) docker-entrypoint.sh 腳本的最后一行, 執(zhí)行的都是 exec "$@"命令

這個(gè)命令的意義在于你已經(jīng)為你的鏡像預(yù)想到了應(yīng)該有的調(diào)用情況, 當(dāng)實(shí)際使用鏡像的人執(zhí)行了你沒有預(yù)料到的可執(zhí)行命令時(shí), 將會(huì)走到腳本的這最后一行, 去執(zhí)行用戶新的可執(zhí)行命令

情況判斷

上面直接說了腳本的最后一行, 在之前的腳本中, 需要充分的去考慮你自己的腳本可能會(huì)被調(diào)用的情況. 還是拿 MySQL 官方的 dockerfile 來說, 他判斷以下情況:

開頭是 - , 認(rèn)為是參數(shù)的情況

開頭是 mysqld, 且用戶 id 為0 (root 用戶) 的情況

開頭是 mysqld 的情況

判斷完自己應(yīng)用的所有調(diào)用形態(tài)之后, 最后應(yīng)該加上exec "$@" 命令兜底

${mysql[@]}

Shell 中的數(shù)組, 直接執(zhí)行 ${mysql[@]} 會(huì)把這個(gè)數(shù)組當(dāng)做可執(zhí)行程序來執(zhí)行

➜ /tmp mysql=( mysql --protocol=socket -uroot -hlocalhost --socket="${SOCKET}" )
➜ /tmp echo ${mysql[1]}
mysql
➜ /tmp echo ${mysql[2]}
--protocol=socket
➜ /tmp echo ${mysql[3]}
-uroot
➜ /tmp echo ${mysql[4]}
-hlocalhost
➜ /tmp echo ${mysql[@]}
mysql --protocol=socket -uroot -hlocalhost --socket=
exec gosu mysql “$BASH_SOURCE” “$@”

這里的 gosu 命令, 是 Linux 中 sudo 命令的輕量級(jí)”替代品”

gosu 是一個(gè) golang 語言開發(fā)的工具, 用來取代 shell 中的 sudo 命令. su 和 sudo 命令有一些缺陷, 主要是會(huì)引起不確定的 TTY, 對信號(hào)量的轉(zhuǎn)發(fā)也存在問題. 如果僅僅為了使用特定的用戶運(yùn)行程序, 使用 su 或 sudo 顯得太重了, 為此 gosu 應(yīng)運(yùn)而生.

gosu 直接借用了 libcontainer 在容器中啟動(dòng)應(yīng)用程序的原理, 使用 /etc/passwd 處理應(yīng)用程序. gosu 首先找出指定的用戶或用戶組, 然后切換到該用戶或用戶組. 接下來, 使用 exec 啟動(dòng)應(yīng)用程序. 到此為止, gosu 完成了它的工作, 不會(huì)參與到應(yīng)用程序后面的聲明周期中. 使用這種方式避免了 gosu 處理 TTY 和轉(zhuǎn)發(fā)信號(hào)量的問題, 把這兩個(gè)工作直接交給了應(yīng)用程序去完成

以上這篇Dockerfile中ENTRYPOINT 和 CMD的區(qū)別說明就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • docker下修改mysql配置文件的方法

    docker下修改mysql配置文件的方法

    這篇文章主要介紹了docker下修改mysql配置文件的方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-06-06
  • Linux環(huán)境下使用Docker搭建Jenkins容器的方法步驟

    Linux環(huán)境下使用Docker搭建Jenkins容器的方法步驟

    本文主要介紹了Linux環(huán)境下使用Docker搭建Jenkins容器的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • Docker配置阿里云鏡像加速pull的實(shí)現(xiàn)

    Docker配置阿里云鏡像加速pull的實(shí)現(xiàn)

    這篇文章主要介紹了Docker配置阿里云鏡像加速pull的實(shí)現(xiàn)操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • Docker Desktop更改鏡像存儲(chǔ)位置的實(shí)現(xiàn)

    Docker Desktop更改鏡像存儲(chǔ)位置的實(shí)現(xiàn)

    本文主要主要介紹了Docker Desktop更改鏡像存儲(chǔ)位置的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • docker安裝influxdb的詳細(xì)教程(性能測試)

    docker安裝influxdb的詳細(xì)教程(性能測試)

    這篇文章主要介紹了docker安裝influxdb的詳細(xì)教程,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • Docker compose安裝部署創(chuàng)建自定義網(wǎng)絡(luò)使用舉例

    Docker compose安裝部署創(chuàng)建自定義網(wǎng)絡(luò)使用舉例

    這篇文章主要為大家介紹了Docker compose安裝部署創(chuàng)建自定義網(wǎng)絡(luò)使用舉例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • windows下Docker部署Flask的詳細(xì)教程

    windows下Docker部署Flask的詳細(xì)教程

    這篇文章主要介紹了windows下Docker部署Flask的教程,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-01-01
  • 解決Jenkins集成docker插件問題的一些方法

    解決Jenkins集成docker插件問題的一些方法

    本文主要介紹了Jenkins集成docker插件問題的一些方法,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • mac下關(guān)于docker的安裝和配置教程

    mac下關(guān)于docker的安裝和配置教程

    在Mac上安裝Docker的詳細(xì)步驟包括系統(tǒng)要求、下載安裝包、安裝DockerDesktop、啟動(dòng)并配置、驗(yàn)證安裝、運(yùn)行HelloWorld容器、管理Docker和卸載Docker
    2024-11-11
  • Docker?Desktop常見的幾種啟動(dòng)失敗問題解決方法

    Docker?Desktop常見的幾種啟動(dòng)失敗問題解決方法

    使用Docker Desktop 可以很方便的啟用 Kubernetes,下面這篇文章主要給大家介紹了關(guān)于Docker?Desktop常見的幾種啟動(dòng)失敗問題解決方法的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-09-09

最新評論