实际工作中,我们可能需要自己去创建一个docker镜像,下面给大家介绍如何创建docker镜像
1. 创建一个最简单的镜像
准备Dockerfile文件
[root@dockhost ~]# mkdir d1 # 创建一个空的目录,避免被别的文件打扰 [root@dockhost ~]# cd d1 # 切换到该目录 [root@dockhost d1]# vi Dockerfile # 创建一个dockerfile文件,名称必须是DockerfileFROM alpine:latestMAINTAINER ybCMD echo 'hello, this is my first image'命令解释 FROM alpine:latest # alpine可以看成是一个基础镜像,它是一个极简版的linux系统。我们自定义的镜像一般都要通过FROM关键字指定一个基础镜像 MAINTAINER yb # 这个指定镜像的所有者,yb是我的名字 CMD echo 'hello, this is my first image' # CMD是command的简写,后面跟实际执行的命令
创建镜像
[root@dockhost d1]# docker build -t hello_docker . #-t后面跟镜像的名字,.表示当前目录下的所有文件Sending build context to Docker daemon 2.048 kBStep 1/3 : FROM alpine:latestTrying to pull repository docker.io/library/alpine ... latest: Pulling from docker.io/library/alpinee7c96db7181b: Pull complete Digest: sha256:769fddc7cc2f0a1c35abb2f91432e8beecf83916c421420e6a6da9f8975464b6Status: Downloaded newer image for docker.io/alpine:latest ---> 055936d39205Step 2/3 : MAINTAINER yb ---> Running in af8fa3cb5325 ---> f728cbd10fa7Removing intermediate container af8fa3cb5325Step 3/3 : CMD echo 'hello, this is my first image' ---> Running in 7ac6db88f533 ---> dd6668981af6Removing intermediate container 7ac6db88f533Successfully built dd6668981af6
运行镜像
[root@dockhost d1]# docker run hello_docker hello, this is my first image查看镜像
[root@dockhost d1]# docker imagesREPOSITORY TAG IMAGE ID CREATED SIZEhello_docker latest dd6668981af6 About a minute ago 5.53 MBdocker.io/alpine latest 055936d39205 2 weeks ago 5.53 MB从docker build的输出和docker images的显示都可以看到在创建镜像hello_docker的同时还下载了基础镜像alpine
2. 将基础镜像换成centos,其余不变。来看有啥变化
[root@dockhost d1]# cd .. # 切到上一层目录
[root@dockhost ~]# mkdir d2 # 创建一个空的目录,避免被别的文件打扰 [root@dockhost ~]# cd d2 # 切换到该目录 [root@dockhost d2]# vi Dockerfile # 创建一个dockerfile文件,名称必须是DockerfileFROM centosMAINTAINER ybCMD echo 'hello, this is my first image'
[root@dockhost d2]# docker build -t hello_docker2 . # 创建镜像,名字改成hello_docker2
省略输出...[root@dockhost d2]# docker run hello_docker2 # 运行镜像
hello, this is my first image[root@dockhost d2]# docker images # 查看镜像
REPOSITORY TAG IMAGE ID CREATED SIZEhello_docker2 latest 1787753b5550 2 minutes ago 202 MBhello_docker latest dd6668981af6 10 minutes ago 5.53 MBdocker.io/alpine latest 055936d39205 2 weeks ago 5.53 MBdocker.io/centos latest 9f38484d220f 2 months ago 202 MB可以看到因为基础镜像的不同,hello_docker与hello_docker2文件尺寸差别好大。具体该选择什么基础镜像呢。 如果是一个单功能的容器,而且自身对alpine比较熟悉,那就用alpine作为基础镜像,否则就使用centos或者ubuntu,根据自己所运行的操作系统来。
3. 接下来创建一个复杂的docker镜像
使用nginx来做测试,从官网下载nginx源码包
[root@dockhost ~]# mdkir mydocker [root@dockhost ~]# cd mydocker 上传安装包到mydocker目录下准备Dockerfile
[root@docker mydocker]# vi Dockerfile# base imageFROM centos# MAINTAINERMAINTAINER yb# put nginx-1.16.0.tar.gz into /usr/local/src and unpack nginxADD nginx-1.16.0.tar.gz /usr/local/src# running required commandRUN yum install -y gcc gcc-c++ glibc make autoconf openssl openssl-devel RUN yum install -y libxslt-devel -y gd gd-devel GeoIP GeoIP-devel pcre pcre-develRUN useradd -M -s /sbin/nologin nginx# mount a dir to containerVOLUME ["/data"]# change dir to /usr/local/src/nginx-1.16.0WORKDIR /usr/local/src/nginx-1.16.0# execute command to compile nginxRUN ./configure --user=nginx --group=nginx --prefix=/usr/local/nginx --with-file-aio --with-http_ssl_module --with-http_realip_module --with-http_addition_module --with-http_xslt_module --with-http_image_filter_module --with-http_geoip_module --with-http_sub_module --with-http_dav_module --with-http_flv_module --with-http_mp4_module --with-http_gunzip_module --with-http_gzip_static_module --with-http_auth_request_module --with-http_random_index_module --with-http_secure_link_module --with-http_degradation_module --with-http_stub_status_module && make && make install# add path ENV PATH /usr/local/nginx/sbin:$PATH# expose portEXPOSE 80# add cmdENTRYPOINT ["nginx"]CMD ["-g","daemon off;"]
创建镜像
docker build -t centos_nginx:v1 .运行镜像
docker run -d -p80:80 centos_nginx:v1打开网页:输入http://10.40.16.61:80/ #其中10.40.16.61我docker运行的主机
命令解释
FROM 选择一个基础镜像 MAINTAINER 镜像作者 ADD 添加文件或目录,类似的命令还有COPY,不过ADD功能更加强大,该命令还能拷贝远程的文件到容器中,比如ftp上的文件,而且能自动解压 RUN 执行命令 VOLUME 给docker添加一个卷组,卷组的含义类似于给操作系统挂载一个磁盘 WORKDIR 切换Docker的目录 ENV 设置环境变量 EXPOSE 暴露端口 ENTRYPOINT 容器入口 CMD 参数,上面例子中ENTRYPOINT+CMD = nginx -g "daemon off;",运行该镜像最后就会执行此条命令文章参考:
建议大家跟着上面的文章走一遍,可以理解很多命令到底是做什么的。我这里贴出一些我看该篇文章的个人疑惑点,以及好的解答。 1.daemon on | off是干什么的? 参考: 意思文章讲的很清楚了,如果想让nginx运行,你必须设置成daemon off2.如何进入到container中?
docker exec -it e09423b0c114 /bin/bash # e09423b0c114是容器id 退出就直接使用exit3.如何查找docker的VOLUME对应主机的目录?
docker inspect e09423b0c114 # e09423b0c114是容器id 返回是一个json字符串,找"Mounts"键 "Mounts": [ { "Type": "volume", "Name": "4d0ff619cee2a21a75af2bea3fe7aa30d4f9fc0bf4612c48c8dced96234cec3e", "Source": "/var/lib/docker/volumes/4d0ff619cee2a21a75af2bea3fe7aa30d4f9fc0bf4612c48c8dced96234cec3e/_data", "Destination": "/data", "Driver": "local", "Mode": "", "RW": true, "Propagation": "" }4.在文章的v7中,应该是先建一个image再运行。但是我运行了之后发现打不开nginx,目前尚不清楚dockerfile里面有onbuild,nginx就不运行,这个先标注,等将来弄明白了再改这段。大家可以忽略这段。
4. Dockerfile语法总结
FROM 指定基础镜像
RUN 执行命令 COPY 添加文件 ADD 添加文件 EXPOSE 暴露端口 WORKDIR 指定路径 MAINTAINER 维护者 ENV 设定环境变量 ENTRYPOINT 容器入口 CMD 执行参数 USER 指定用户 VOLUMN 添加卷组