Dockerfile中的RUN,CMD,ENTRTPOINT三個指令均可以用來指明容器中所運行的指令,但這三者存在的細微的區別。
簡單來說:
RUN
RUN指令一般用於在容器內安裝軟件包或者是執行其他的命令,如
RUN yum install -y telnet
RUN touch web.xml
CMD
CMD指令主要用來指明生成的Docker鏡像在啟動時的命令及參數,這個指令可以被docker run后面的命令所取代,比如下面這個Dockerfile文件
FROM busybox
CMD echo "hello world"
CMD指明了Docker鏡像在運行時的輸出一個"hello world"
[root@bochs Docker]# docker build -t test .
Sending build context to Docker daemon 2.048kB
Step 1/2 : FROM busybox
---> 83aa35aa1c79
Step 2/2 : CMD echo "hello world"
---> Running in a1a4d74137d2
Removing intermediate container a1a4d74137d2
---> 651b45b58fe9
Successfully built 651b45b58fe9
Successfully tagged test:latest
[root@bochs Docker]# docker run -it test
hello world
但是如果在docker run后添加其他指令。那么CMD將直接被替換
[root@bochs Docker]# docker run -it test ls
bin dev etc home proc root sys tmp usr var
ENTRYPOINT
ENTRYPOINT與CMD類似,區別在於ENTRYPOINT一定會被執行。如果一個Dockerfile中同時存在ENTRYPOINT和
CMD,CMD中的參數會被當做額外參數傳給ENTRYPOINT。
[root@bochs Docker]# cat Dockerfile
FROM busybox
ENTRYPOINT ["/bin/echo","hello"]
CMD ["world"]
通過docker run 來運行,CMD變成了ENTRYPOINT的參數:
[root@bochs Docker]# docker run -it test2
hello world
但是如果指明docker run 的參數china,那么輸出就會變為:
[root@bochs Docker]# docker run -it test2 china
hello china
原本CMD中帶的參數world被docker run中的china所替換,但ENTRYPOINT自帶的hello依然正常輸出
Shell與Exec格式
CMD,RUN,ENTRYPOINT可以用兩種格式來傳遞命令和參數,Shell一般表示為指令+命令,如:
RUN yum install -y telnet
CMD echo "hello world"
第一個大寫的單詞是Dockerfile的指令。后面跟的就是命令,可以拿到shell中單獨執行
Exec格式可以表示為:指令+["命令","命令參數1","命令參數2",...],比如:
RUN ["yum","install","telnet"]
ENTRYPOINT ["/bin/bash","-c","echo hello world"]
對於這兩種格式來說,CMD和ENTRYPOINT最好使用Exec格式,命令和參數分開,層次性較強,而RUN則都可以。
注意:ENTRYPOINT的Shell格式和Exec格式差異很大
比如下面這個Shell格式的ENTRYPOINT
FROM busybox
ENTRYPOINT echo "hello"
CMD "world"
在運行所生成的容器時,僅會輸出hello,而CMD帶的"world"會被忽略。同樣的docker run帶的參數也同樣會被忽略
[root@bochs Docker]# docker run -it test
hello
[root@bochs Docker]# docker run -it test china
hello