首先應該了解Docker鏡像的分層機制,這個網上文章很多,簡單說就是對鏡像的每次修改都是在原鏡像基礎上加了一層包裝;
構建自己的docker鏡像有兩種方式;
一種是在啟動容器后,直接對容器進行操作,然后commit,好處是方便直觀,但會導致很多中間層,且不好追溯,所以不推薦這樣用;
另一種是編輯dockerfile,然后通過docker build生成鏡像,下面詳細說說如何使用dockerfile構建一個帶jdk的centos鏡像;
先準備一個文件夾img,把需要的文件放到子文件夾resource中,這里還拿了一個rpm文件演示如何安裝,如下:
img/
├── Dockerfile
└── resource
├── jdk1.7.0_65
├── nc-1.84-22.el6.x86_64.rpm
文件Dockerfile的內容如下:
FROM daocloud.io/centos:latest COPY resource /usr/local RUN cd /usr/local && rpm -ivh nc-1.84-22.el6.x86_64.rpm && yum install which -y && rm -rf nc-1.84-22.el6.x86_64.rpm && yum clean all ENV JAVA_HOME=/usr/local/jdk1.7.0_65/jre ENV CLASSPATH=.:$JAVA_HOME/lib PATH=$JAVA_HOME/bin:$PATH
FROM 必須位于第一行,表示基礎鏡像,這里為centos,若不需要任何基礎環境就用scratch;
COPY 拷貝宿主機的資源到鏡像,這里是將resource下所有東西拷貝到鏡像的/usr/local/目錄;
RUN 鏡像中執行命令,這里安裝了一個rpm包和yum安裝which命令,然后進行了清理;
這里把多個命令合并到一個RUN語句是為了減少層數;
ENV 設置鏡像的環境變量,由于后面兩個變量依賴JAVA_HOME,所以必須分到兩個ENV語句了;
其他命令還有:ADD/CMD/ENTRYPOINT/VOLUME/ARG/EXPOSE/WORKDIR/USER/HEALTHCHECK/ONBUILD等;
下面構建鏡像:
# cd img/ # docker build -t my/base . # docker images REPOSITORY TAG IMAGE ID CREATED SIZE my/base latest 2cdd09a142e5 24 seconds ago 359MB daocloud.io/centos latest a8493f5f50ff 4 days ago 192MB
. 命令最后傳入的是build的工作目錄,這個工作目錄是很重要的;
工作目錄里所有的文件會傳輸到docker服務端,而且dockfile中指定文件時(COPY那行的resource)必須是工作目錄的相對路徑;
所以,在img目錄外面這樣build也是一樣的:
docker build -t my/base img/
-t 指定鏡像在倉庫的名字;
-f 指定dockfile,因為我們這里用的默認名稱Dockerfile,所以不用指定;
--network=host build也是啟動了一個鏡像執行的,因為默認的bridge網絡模式有網絡損耗,所以如果build中包含較多外網訪問,建議改為host:
docker build --network=host -t my/base .
下面啟動測試下這個鏡像:
# docker run --name mb -dti my/base c86b648c10174271e080f90e8d437a4ee2e25993003bdc68317477514349cb10 # docker attach mb [root@c86b648c1017 /]# which java /usr/local/jdk1.7.0_65/jre/bin/java [root@c86b648c1017 /]# java -version java version "1.7.0_65" Java(TM) SE Runtime Environment (build 1.7.0_65-b17) Java HotSpot(TM) 64-Bit Server VM (build 24.65-b04, mixed mode)