实际工作中,我们可能需要自己去创建一个docker镜像,下面给大家介绍如何创建docker镜像
1. 创建一个最简单的镜像
准备Dockerfile文件
[root@dockhost ~]# mkdir d1 # 创建一个空的目录,避免被别的文件打扰
[root@dockhost ~]# cd d1 # 切换到该目录
[root@dockhost d1]# vi Dockerfile # 创建一个dockerfile文件,名称必须是Dockerfile
FROM alpine:latest
MAINTAINER yb
CMD 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 kB
Step 1/3 : FROM alpine:latest
Trying to pull repository docker.io/library/alpine ...
latest: Pulling from docker.io/library/alpine
e7c96db7181b: Pull complete
Digest: sha256:769fddc7cc2f0a1c35abb2f91432e8beecf83916c421420e6a6da9f8975464b6
Status: Downloaded newer image for docker.io/alpine:latest
---> 055936d39205
Step 2/3 : MAINTAINER yb
---> Running in af8fa3cb5325
---> f728cbd10fa7
Removing intermediate container af8fa3cb5325
Step 3/3 : CMD echo 'hello, this is my first image'
---> Running in 7ac6db88f533
---> dd6668981af6
Removing intermediate container 7ac6db88f533
Successfully built dd6668981af6
运行镜像
[root@dockhost d1]# docker run hello_docker
hello, this is my first image
查看镜像
[root@dockhost d1]# docker images
REPOSITORY TAG IMAGE ID CREATED SIZE
hello_docker latest dd6668981af6 About a minute ago 5.53 MB
docker.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文件,名称必须是Dockerfile
FROM centos
MAINTAINER yb
CMD 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 SIZE
hello_docker2 latest 1787753b5550 2 minutes ago 202 MB
hello_docker latest dd6668981af6 10 minutes ago 5.53 MB
docker.io/alpine latest 055936d39205 2 weeks ago 5.53 MB
docker.io/centos latest 9f38484d220f 2 months ago 202 MB
可以看到因为基础镜像的不同,hello_docker与hello_docker2文件尺寸差别好大。具体该选择什么基础镜像呢。
如果是一个单功能的容器,而且自身对alpine比较熟悉,那就用alpine作为基础镜像,否则就使用centos或者ubuntu,根据自己所运行的操作系统来。
3. 接下来创建一个复杂的docker镜像
使用nginx来做测试,从官网下载nginx源码包 http://nginx.org/en/download.html
[root@dockhost ~]# mdkir mydocker
[root@dockhost ~]# cd mydocker
上传安装包到mydocker目录下
准备Dockerfile
[root@docker mydocker]# vi Dockerfile
# base image
FROM centos # MAINTAINER
MAINTAINER yb # put nginx-1.16.0.tar.gz into /usr/local/src and unpack nginx
ADD nginx-1.16.0.tar.gz /usr/local/src # running required command
RUN 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-devel
RUN useradd -M -s /sbin/nologin nginx # mount a dir to container
VOLUME ["/data"] # change dir to /usr/local/src/nginx-1.16.0
WORKDIR /usr/local/src/nginx-1.16.0 # execute command to compile nginx
RUN ./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 port
EXPOSE 80 # add cmd
ENTRYPOINT ["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;",运行该镜像最后就会执行此条命令
文章参考:https://www.cnblogs.com/jsonhc/p/7767669.html
建议大家跟着上面的文章走一遍,可以理解很多命令到底是做什么的。我这里贴出一些我看该篇文章的个人疑惑点,以及好的解答。
1.daemon on | off是干什么的?
参考:https://www.cnblogs.com/weifeng1463/p/10277178.html
意思文章讲的很清楚了,如果想让nginx运行,你必须设置成daemon off
2.如何进入到container中?
docker exec -it e09423b0c114 /bin/bash # e09423b0c114是容器id
退出就直接使用exit
3.如何查找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 添加卷组