在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

信號的理念以及Dockerfile中ENTRYPOINT和CMD指令

馬哥Linux運維 ? 來源:CSDN技術社區 ? 作者:CSDN技術社區 ? 2022-06-12 16:42 ? 次閱讀
最近把 Docker 官方的 Docker Reference 文檔又讀了一遍,發現有些細節深究起來,還是有很多可挖的。針對寫 Dockerfile ,大部分時候只要照葫蘆畫瓢,基本也不會有什么大的問題,但是如果再深入理解一下那就更有意思了。要說如何優雅的關閉容器,那就不得不提到信號(Signal)的理念,以及 Dockerfile 中 ENTRYPOINT 和 CMD 指令了。在具體說優雅關閉之前,先了解一下信號這個 Linux 中的基礎概念。

1 信號

信號是事件發生時對進程的通知機制,有時也稱之為軟件中斷。

信號有不同的類型,Linux 對標準信號的編號為 1~31,可以通過 kill -l 獲取信號名稱:
# kill -l 1) SIGHUP       2) SIGINT       3) SIGQUIT       4) SIGILL       5) SIGTRAP      6) SIGABRT       7) SIGBUS       8) SIGFPE       9) SIGKILL 10) SIGUSR1    11) SIGSEGV     12) SIGUSR2 13) SIGPIPE    14) SIGALRM     15) SIGTERM... ...
實際列出的信號超過了 31 個,有些是其它名稱的同義詞,有些則是定義但未使用的。以下介紹幾個常用的信號:
  • 1) SIGHUP當終端斷開(掛機)時,將發送該信號給終端控制進程。SIGHUP 信號還可用于守護進程(比如,init 等)。許多守護進程會在收到 SIGHUP 信號時重新進行初始化并重讀配置文件。

  • 2) SIGINT當用戶鍵入終端中斷字符(通常為Control-C) 時,終端驅動程序將發送該信號給前臺進程組。該信號的默認行為是終止進程。

  • 3) SIGQUIT當用戶在鍵盤上鍵入退出字符(通常為Control-)時,該信號將發往前臺進程組。默認情況下,該信號終止進程,并生成用于調試的核心轉儲文件。進程如果陷入無限循環,或者不再響應時,使用 SIGQUIT 信號就很合適。

  • 9) SIGKILL此信號為 “必殺(sure kill)” 信號,處理器程序無法將其阻塞、忽略或者捕獲,故而 “一擊必殺”,總能終止程序。

  • 15) SIGTERM這是用來終止進程的標準信號,也是killkillallpkill命令所發送的默認信號。精心設計的應用程序應當為 SIGTERM 信號設置處理器程序,以便其能夠預先清除臨時文件和釋放其它資源,從而全身而退。因此,總是應該先嘗試使用SIGTERM信號來終止進程,而把SIGKILL作為最后手段,去對付那些不響應SIGTERM信號的失控進程。

  • 20) SIGTSTP這是作業控制的停止信號,當用戶在鍵盤上輸入掛起字符(通常為 Control-Z )時,將該信號給前臺進程組,使其停止運行。

值得注意的是,Control-D不會發起信號,它表示 EOF(End-Of-File),關閉標準輸入(stdin)管道(比如可以通過Control-D退出當前 shell)。如果程序不讀取當前輸入的話,是不受 Control-D 影響的。

程序可以針對信號捕捉,然后執行相應函數:

51df66c8-de8c-11ec-ba43-dac502259ad0.png

以上知識大部分都來自 《Linux/UNIX 系統編程手冊》,想要了解更多的,可以查看該書上冊的 20、21、22 章節。

2 ENTRYPOINT 、 CMD

可能有人會問,說了半天,那信號和優雅的關閉容器有半毛錢的關系啊?話說,這和錢確實沒關系,但是和如何優雅關閉容器卻關系密切。接著說 Dockerfile 中的 ENTRYPOINT 和 CMD 指令,它們的主要功能是指定容器啟動時執行的程序。

CMD有三種格式:

  • CMD ["executable","param1","param2"](exec 格式, 推薦使用這種格式)

  • CMD ["param1","param2"](作為 ENTRYPOINT 指令參數

  • CMD command param1 param2(shell 格式,默認 /bin/sh -c )

ENTRYPOINT有兩種格式:

  • ENTRYPOINT ["executable", "param1", "param2"](exec 格式,推薦優先使用這種格式)

  • ENTRYPOINT command param1 param2(shell 格式)

其中,不管你Dockerfile用其中哪個指令,兩個指令都推薦使用 exec 格式,而不是 shell 格式。原因就是因為使用 shell 格式之后,程序會以/bin/sh -c的子命令啟動,并且 shell 格式下不會傳遞任何信號給程序。這也就導致,在docker stop容器的時候,以這種格式運行的程序捕捉不到發送的信號,也就談不上優雅的關閉了。

?  ~ docker stop --help
Usage:  docker stop [OPTIONS] CONTAINER [CONTAINER...]
Stop one or more running containers
Options:      --help       Print usage  -t, --time int   Seconds to wait for stop before killing it (default 10)
docker stop停掉容器的時候,默認會發送一個SIGTERM的信號,默認 10s 后容器沒有停止的話,就SIGKILL強制停止容器。通過-t選項可以設置等待時間。
?  ~ docker kill --help
Usage:  docker kill [OPTIONS] CONTAINER [CONTAINER...]
Kill one or more running containers
Options:      --help            Print usage  -s, --signal string   Signal to send to the container (default "KILL")

	

通過 docker kill-s選項還可以指定給容器發送的信號。

所以,說了那么多,只要Dockerfile中通過 exec 格式執行容器啟動命令就相安無事了?那當然是,沒有那么簡單的了,接下來我們通過實例來看看具體的效果是怎么樣的。

3 實例

通過 Go 寫一個簡單的信號處理器:

?  ~ cat signals.gopackage main
import (    "fmt"    "os"    "os/signal"    "syscall")
func main() {    sigs := make(chan os.Signal, 1)    done := make(chan bool, 1)
    signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
    go func() {        sig := <-sigs        fmt.Println()        fmt.Println(sig)        done <- true    }()
    fmt.Println("awaiting signal")    <-done    fmt.Println("exiting")}

3.1 實例 1

?  ~ GOOS=linux GOARCH=amd64 go build signals.go?  ~ lsDockerfile signals    signals.go?  ~ cat DockerfileFROM busybox
COPY signals /signals
CMD ["/signals"]    # exec 格式執行?  ~ docker build -t signals .
通過 tmux 開啟兩個面板,一個運行容器,一個執行 docker stop :
?  ~ docker run -it --rm --name signals signalsawaiting signal
terminatedexiting
?  ~ time docker stop signalssignalsdocker stop signals  0.01s user 0.02s system 4% cpu 0.732 total?  ~

可以發現,容器停止之前,程序接收到信號并輸出相應信息,并且停止總耗時為 0.732 s,達到了優雅的效果。

修改 Dockerfile 中 CMD 執行格式,執行相同操作:

?  ~ cat DockerfileFROM busybox
COPY signals /signals
CMD /signals        # shell 格式執行?  ~ docker build -t signals .
?  ~ docker run -it --rm --name signals signalsawaiting signal?  ~
?  ~ time docker stop signalssignalsdocker stop signals  0.01s user 0.01s system 0% cpu 10.719 total

通過 shell 格式之后,可以發現容器停止之前,程序并未接收到任何信號,并且停止時間為 10.719s,說明該容器是被強制停止的。

結論很明顯,為了優雅的退出容器,我們應該采用 exec 這種格式。

3.2 實例 2

通過實例 1 我們都會在 Dockerfile 中都會通過 exec 這種格式來執行程序了,那如果執行的程序本身也是一個 shell 腳本呢?
?  ~ lsDockerfile signals    signals.go start.sh?  ~ cat DockerfileFROM busybox
COPY signals /signalsCOPY start.sh /start.sh     # 引入 shell 腳本啟動
CMD ["/start.sh"]?  ~ cat start.sh#!/bin/sh
/signals?  ~

測試依然引用實例 1 中的方法:

?  ~ docker run -it --rm --name signals signalsawaiting signal?  ~
?  ~ time docker stop signalssignalsdocker stop signals  0.01s user 0.02s system 0% cpu 10.765 total?  ~
可以發現,即使 Dockerfile 中的 CMD 指令使用的是 exec 格式,容器中的程序依然沒有接收到信號,最后被強制關閉。因為 shell 腳本中執行的原因,導致信號依然沒有被傳遞,我們需要針對 shell 腳本做一些改造:
?  ~ cat start.sh#!/bin/sh
exec /signals   # 加入 exec 執行?  ~ docker build -t signals .
?  ~ docker run -it --rm --name signals signalsawaiting signal
terminatedexiting
?  ~ time docker stop signalssignalsdocker stop signals  0.02s user 0.02s system 4% cpu 0.744 total?  ~

可以看到,加入 exec 命令之后,程序又可以接收到信號正常退出了。當然,如果你 Dockerfile 中的 CMD 是以 shell 格式運行的,即使啟動腳本中加入 exec 也是無效的。再者,如果你的程序本身不能針對信號做一些處理,也就談不上優雅關閉了。

原文標題:如何優雅的關閉容器

文章出處:【微信公眾號:馬哥Linux運維】歡迎添加關注!文章轉載請注明出處。

審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11357

    瀏覽量

    210828
  • 信號
    +關注

    關注

    11

    文章

    2812

    瀏覽量

    77229
  • 容器
    +關注

    關注

    0

    文章

    499

    瀏覽量

    22153
  • Docker
    +關注

    關注

    0

    文章

    492

    瀏覽量

    12029

原文標題:如何優雅的關閉容器

文章出處:【微信號:magedu-Linux,微信公眾號:馬哥Linux運維】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    11.Dockerfile配置之CMDENTRYPOINT

    前端編程語言
    電子學習
    發布于 :2023年02月01日 21:52:51

    Dockerfile的最佳實踐

    ”微服務一條龍“最佳指南-“最佳實踐”篇:Dockerfile
    發表于 07-11 16:22

    Dockerfile使用規則

    Dockerfile編寫規范
    發表于 08-12 14:30

    CMD文件定義怎么理解?

    ,256k L2 RAM, 以及通過EMIFA及DDR2/mDDR擴展的RAM或FLASH,這些空間可供用戶使用;CMD文件即將這些空間進行劃分。疑問如下:1. 上述CMD的定義,EntryP
    發表于 10-22 11:57

    一文詳解DockerFile基礎知識

    # 當構建一個被繼承Dockerfile 這個時候運行ONBUILD指定,觸發指令COPY # 將文件拷貝到鏡像ENV # 構建的時候設置環境變量構建自己的Centos鏡像Docker hub
    發表于 09-15 15:54

    DSP28335—CMD文件的各個段解釋

    cmd以下代碼如何解釋?請看本文
    發表于 12-26 17:26 ?18次下載

    系統之上的進程級別虛擬化技術——Docker

    的正常代碼的區別。而且后者在一個 Dockerfile 文件只能有一個存在。CMD/ENTRYPOINT 的區別除了在寫法上有區別之外,還有在docker run命令后增加
    的頭像 發表于 05-02 14:55 ?5107次閱讀
    系統之上的進程級別虛擬化技術——Docker

    全面詳解Dockerfile文件

    Docker 可以通過讀取 Dockerfile 指令自動構建鏡像。Dockerfile 是一個文本文檔,其中包含了用戶創建鏡像的所有命令和說明。 一、 變量 變量用
    的頭像 發表于 09-22 15:38 ?1950次閱讀

    鏡像構建Dockerfile的介紹

    Dockerfile 是一個用來構建鏡像的文本文件,文本內容包含了一條條構建鏡像所需的指令和說明。
    的頭像 發表于 09-06 09:36 ?1228次閱讀

    Dockerfile的最佳實踐

    隨著應用的容器化、上云后,將伴隨著 Docker 鏡像的構建,構建 Docker 鏡像成為了最基本的一步,其中 Dockerfile 便是用來構建鏡像的一種文本文件,鏡像的優劣全靠
    的頭像 發表于 01-20 10:59 ?1054次閱讀
    <b class='flag-5'>Dockerfile</b>的最佳實踐

    使用匿名管道技術獲取CMD命令的執行結果

    遠程 CMD 是指惡意程序接收到控制端發送的 CMD 指令后,在本地執行 CMD 命令,并將執行結果回傳至控制端。本文將演示使用匿名管道技術獲取 C
    的頭像 發表于 04-03 18:04 ?4211次閱讀

    Dockerfile定義Docker鏡像的構建過程

    了解Dockerfile Dockerfile 是一個文本文件,用于定義 Docker 鏡像的構建過程。它以指令的形式描述了如何構建鏡像,從基礎鏡像開始逐步添加配置、文件和依賴,最終形成我們所需
    的頭像 發表于 09-30 10:22 ?2637次閱讀

    如何使用dockerfile創建鏡像

    如何使用Dockerfile創建鏡像,包括Dockerfile的語法和常用指令以及具體操作步驟。 編寫Dockerfile
    的頭像 發表于 11-23 09:52 ?831次閱讀

    提升DevOps效率,從基礎到進階的Dockerfile編寫技巧

    目錄 Dockerfile 基本結構 指令 創建鏡像(centos版) 創建鏡像(alpine版) 基本結構 Dockerfile 是一個文本格式的配置文件,用戶可以使用 Dockerfile
    的頭像 發表于 11-26 09:44 ?159次閱讀
    提升DevOps效率,從基礎到進階的<b class='flag-5'>Dockerfile</b>編寫技巧

    Dockerfile鏡像制作與Docker-Compose容器編排

    Dockerfile鏡像制作 docker/podman, 鏡像是容器的基礎,每次執行docker run的時候都會指定哪個基本鏡像作為容器運行的基礎。我們之前的docker的操作都是使用來
    的頭像 發表于 01-07 11:01 ?226次閱讀
    <b class='flag-5'>Dockerfile</b>鏡像制作與Docker-Compose容器編排
    主站蜘蛛池模板: 国产亚洲精品线观看77 | 天天插天天狠天天透 | 久久久夜 | 免费观看黄a一级视频 | 久久久久久国产精品免费 | 日韩精品一区二区三区毛片 | 天天se天天cao综合网蜜芽 | 亚洲最大黄色网址 | 免费色黄网站 | www.毛片网站 | 视频在线观看一区二区 | 天堂最新版中文网 | 免费看大尺度视频在线观看 | 天天射天天拍 | 色中色软件| 日本黄色片在线播放 | 国产在线永久视频 | 天堂资源在线种子资源 | 欧美一区二区三区激情啪啪 | 长腿丝袜美女被啪啪 | 看视频免费网站 | 午夜影院0606免费 | 色婷婷777| 色中色综合网 | 日本亚洲精品成人 | 午夜免费视频观看在线播放 | www.丁香.com| 日本不卡免费高清一级视频 | 欧洲成品大片在线播放 | 美女好紧好大好爽12p | 91精品国产免费久久久久久青草 | 1024你懂的在线观看 | 午夜在线观看免费高清在线播放 | 毛片黄| 国产欧美日韩va | 天天色狠狠干 | 国产精品不卡片视频免费观看 | 欧美两性网 | 国产一级特黄的片子 | 国产卡一卡2卡三卡免费视频 | 男男gay污小黄文 |