docker创建镜像的两种方法

本文详细介绍了在Docker中创建和管理镜像的方法,包括通过DockerHub获取镜像并更新,利用本地文件系统创建镜像,使用Dockerfile进行镜像构建,以及如何上传镜像到仓库。同时,还讲解了如何使用docker commit和docker build命令,以及如何通过Dockerfile添加应用程序和配置。

创建镜像有很多方法,用户可以从 Docker Hub 获取已有镜像并更新,也可以利用本地文件系统创建一个。

修改已有镜像

先使用下载的镜像启动容器。

$ sudo docker run -t -i training/sinatra /bin/bash
root@0b2616b0e5a8:/#

注意:记住容器的 ID,稍后还会用到。

在容器中添加 json 和 gem 两个应用。

root@0b2616b0e5a8:/# gem install json

当结束后,我们使用 exit 来退出,现在我们的容器已经被我们改变了,使用 docker commit 命令来提交更新后的副本。

$ sudo docker commit -m "Added json gem" -a "Docker Newbee" 0b2616b0e5a8 ouruser/sinatra:v2
4f177bd27a9ff0f6dc2a830403925b5360bfe0b93d476f7fc3231110e7f71b1c

其中,-m 来指定提交的说明信息,跟我们使用的版本控制工具一样;-a 可以指定更新的用户信息;之后是用来创建镜像的容器的 ID;最后指定目标镜像的仓库名和 tag 信息。创建成功后会返回这个镜像的 ID 信息。

使用 docker images 来查看新创建的镜像。

$ sudo docker images
REPOSITORY          TAG     IMAGE ID       CREATED       VIRTUAL SIZE
training/sinatra    latest  5bc342fa0b91   10 hours ago  446.7 MB
ouruser/sinatra     v2      3c59e02ddd1a   10 hours ago  446.7 MB
ouruser/sinatra     latest  5db5f8471261   10 hours ago  446.7 MB

之后,可以使用新的镜像来启动容器

$ sudo docker run -t -i ouruser/sinatra:v2 /bin/bash
root@78e82f680994:/#

利用 Dockerfile 来创建镜像

使用 docker commit 来扩展一个镜像比较简单,但是不方便在一个团队中分享。我们可以使用 docker build 来创建一个新的镜像。为此,首先需要创建一个 Dockerfile,包含一些如何创建镜像的指令。

新建一个目录和一个 Dockerfile

$ mkdir sinatra
$ cd sinatra
$ touch Dockerfile

Dockerfile 中每一条指令都创建镜像的一层,例如:

# This is a comment
FROM ubuntu:14.04
MAINTAINER Docker Newbee <newbee@docker.com>
RUN apt-get -qq update
RUN apt-get -qqy install ruby ruby-dev
RUN gem install sinatra

Dockerfile 基本的语法是

  • 使用#来注释
  • FROM 指令告诉 Docker 使用哪个镜像作为基础
  • 接着是维护者的信息
  • RUN开头的指令会在创建中运行,比如安装一个软件包,在这里使用 apt-get 来安装了一些软件

编写完成 Dockerfile 后可以使用 docker build 来生成镜像。

$ sudo docker build -t="ouruser/sinatra:v2" .
Uploading context  2.56 kB
Uploading context
Step 0 : FROM ubuntu:14.04
 ---> 99ec81b80c55
Step 1 : MAINTAINER Newbee <newbee@docker.com>
 ---> Running in 7c5664a8a0c1
 ---> 2fa8ca4e2a13
Removing intermediate container 7c5664a8a0c1
Step 2 : RUN apt-get -qq update
 ---> Running in b07cc3fb4256
 ---> 50d21070ec0c
Removing intermediate container b07cc3fb4256
Step 3 : RUN apt-get -qqy install ruby ruby-dev
 ---> Running in a5b038dd127e
Selecting previously unselected package libasan0:amd64.
(Reading database ... 11518 files and directories currently installed.)
Preparing to unpack .../libasan0_4.8.2-19ubuntu1_amd64.deb ...
Setting up ruby (1:1.9.3.4) ...
Setting up ruby1.9.1 (1.9.3.484-2ubuntu1) ...
Processing triggers for libc-bin (2.19-0ubuntu6) ...
 ---> 2acb20f17878
Removing intermediate container a5b038dd127e
Step 4 : RUN gem install sinatra
 ---> Running in 5e9d0065c1f7
. . .
Successfully installed rack-protection-1.5.3
Successfully installed sinatra-1.4.5
4 gems installed
 ---> 324104cde6ad
Removing intermediate container 5e9d0065c1f7
Successfully built 324104cde6ad

其中 -t 标记来添加 tag,指定新的镜像的用户信息。 “.” 是 Dockerfile 所在的路径(当前目录),也可以替换为一个具体的 Dockerfile 的路径。

可以看到 build 进程在执行操作。它要做的第一件事情就是上传这个 Dockerfile 内容,因为所有的操作都要依据 Dockerfile 来进行。 然后,Dockfile 中的指令被一条一条的执行。每一步都创建了一个新的容器,在容器中执行指令并提交修改(就跟之前介绍过的 docker commit 一样)。当所有的指令都执行完毕之后,返回了最终的镜像 id。所有的中间步骤所产生的容器都被删除和清理了。

*注意一个镜像不能超过 127 层

此外,还可以利用 ADD 命令复制本地文件到镜像;用 EXPOSE 命令来向外部开放端口;用 CMD 命令来描述容器启动后运行的程序等。例如

# put my local web site in myApp folder to /var/www
ADD myApp /var/www
# expose httpd port
EXPOSE 80
# the command to run
CMD ["/usr/sbin/apachectl", "-D", "FOREGROUND"]

现在可以利用新创建的镜像来启动一个容器。

$ sudo docker run -t -i ouruser/sinatra:v2 /bin/bash
root@8196968dac35:/#

还可以用 docker tag 命令来修改镜像的标签。

$ sudo docker tag 5db5f8471261 ouruser/sinatra:devel
$ sudo docker images ouruser/sinatra
REPOSITORY          TAG     IMAGE ID      CREATED        VIRTUAL SIZE
ouruser/sinatra     latest  5db5f8471261  11 hours ago   446.7 MB
ouruser/sinatra     devel   5db5f8471261  11 hours ago   446.7 MB
ouruser/sinatra     v2      5db5f8471261  11 hours ago   446.7 MB

从本地文件系统导入

要从本地文件系统导入一个镜像,可以使用 openvz(容器虚拟化的先锋技术)的模板来创建: openvz 的模板下载地址为 templates 。

比如,先下载了一个 ubuntu-14.04 的镜像,之后使用以下命令导入:

sudo cat ubuntu-14.04-x86_64-minimal.tar.gz  |docker import - ubuntu:14.04

然后查看新导入的镜像。

docker images
REPOSITORY          TAG                 IMAGE ID            CREATED             VIRTUAL SIZE
ubuntu              14.04               05ac7c0b9383        17 seconds ago      215.5 MB

上传镜像

用户可以通过 docker push 命令,把自己创建的镜像上传到仓库中来共享。例如,用户在 Docker Hub 上完成注册后,可以推送自己的镜像到仓库中。

$ sudo docker push ouruser/sinatra
The push refers to a repository [ouruser/sinatra] (len: 1)
Sending image list
Pushing repository ouruser/sinatra (3 tags)
<think> 我们正在讨论如何创建Docker镜像。根据引用内容,创建Docker镜像主要有两种方法: 1. 使用Dockerfile构建镜像(引用[1]和引用[2]) 2. 通过容器创建镜像(引用[3]) 下面我将分别解释这两种方法的具体命令: ### 方法一:使用Dockerfile构建镜像(推荐) 这是自动化构建镜像的标准方法,需要编写一个Dockerfile文件,然后使用`docker build`命令构建镜像。 **步骤:** 1. 创建Dockerfile文件,编写构建指令。例如: ```dockerfile # 指定基础镜像 FROM ubuntu:20.04 # 运行命令安装软件 RUN apt-get update && apt-get install -y nginx # 复制文件 COPY index.html /var/www/html/ # 设置环境变量 ENV HOST_NAME my_server # 暴露端口 EXPOSE 80 # 设置启动命令 CMD ["nginx", "-g", "daemon off;"] ``` 2. 在Dockerfile所在目录执行构建命令: ```bash docker build -t my-nginx-image:latest . ``` - `-t`:指定镜像的名称和标签,格式为`name:tag` - `.`:表示Dockerfile所在的当前目录 ### 方法二:通过容器创建镜像 这种方法基于一个已存在的容器创建镜像。适用于对容器进行修改后保存为镜像的场景。 **步骤:** 1. 启动一个容器(如果已有正在运行的容器,可跳过此步): ```bash docker run -it --name mycontainer ubuntu:20.04 /bin/bash ``` 2. 进入容器并进行修改(如安装软件、修改配置等): ```bash docker exec -it mycontainer bash # 在容器内执行操作,例如安装nginx apt-get update && apt-get install -y nginx ``` 3. 退出容器,并将容器提交为新的镜像: ```bash docker commit -m "Installed nginx" -a "Your Name" mycontainer my-nginx-image:latest ``` - `-m`:提交信息,描述修改内容 - `-a`:作者信息 - `mycontainer`:容器名称或ID - `my-nginx-image:latest`:新镜像的名称和标签 ### 两种方法对比 | 方法 | 适用场景 | 优点 | 缺点 | |--------------------|----------------------------------|--------------------------|--------------------------| | Dockerfile构建 | 自动化构建、版本控制、持续集成 | 可重复构建、易于版本管理 | 编写Dockerfile需要学习 | | 容器提交 | 临时修改、快速保存状态 | 操作简单、快速 | 难以追溯变更、不易自动化 | ### 补充:镜像导入导出(引用[4]) - **导出镜像**(保存为文件): ```bash docker save -o my-nginx-image.tar my-nginx-image:latest ``` - **导入镜像**: ```bash docker load -i my-nginx-image.tar ``` 根据您的需求,如果您希望自动化、可重复地构建镜像,推荐使用Dockerfile方法[^1][^2]。如果只是临时修改容器并保存状态,则可以使用容器提交方法[^3]。 §§相关问题§§ 1. Dockerfile中的指令执行顺序有什么讲究? 2. 如何优化Docker镜像的体积? 3. 使用`docker commit`创建镜像Dockerfile创建镜像有何本质区别? 4. 如何将现有的容器状态保存为Dockerfile? </think> ### Docker 创建镜像两种主要方法 根据 Docker 官方文档和最佳实践[^1][^2][^3],创建 Docker 镜像主要有以下两种方法: --- #### 方法一:使用 Dockerfile 构建镜像(推荐) 这是最标准的镜像创建方式,通过编写 Dockerfile 文件定义构建步骤,然后使用 `docker build` 命令创建镜像。 ##### 操作步骤: 1. **创建 Dockerfile 文件**: ```dockerfile # 指定基础镜像 [^1] FROM ubuntu:20.04 # 运行命令安装依赖 [^1] RUN apt-get update && apt-get install -y nginx # 复制文件到镜像 [^1] COPY index.html /var/www/html/ # 设置环境变量 [^1] ENV HOST_NAME=my_server # 声明暴露端口 [^1] EXPOSE 80 # 设置容器启动命令 [^1] CMD ["nginx", "-g", "daemon off;"] ``` 2. **构建镜像**: ```bash docker build -t my-nginx-image:latest . ``` - `-t`:指定镜像名称和标签 - `.`:表示 Dockerfile 所在目录 --- #### 方法二:通过容器创建镜像(修改现有容器) 该方法适用于基于现有容器修改后创建镜像[^3]: ##### 操作步骤: 1. **启动容器并修改**: ```bash docker run -it --name temp_container ubuntu:20.04 /bin/bash # 在容器内执行修改操作,例如: apt-get update && apt-get install -y nginx ``` 2. **提交容器为镜像**: ```bash docker commit -m "Added Nginx" -a "YourName" temp_container my-nginx-image:latest ``` - `-m`:提交信息 - `-a`:作者信息 - `temp_container`:容器名或ID - `my-nginx-image:latest`:新镜像名称 --- ### 两种方法对比 | 方法 | 适用场景 | 优点 | 缺点 | |--------------------|----------------------------------|--------------------------|--------------------------| | Dockerfile构建 | 自动化构建、版本控制、持续集成 | 可重复构建、易于版本管理 | 编写Dockerfile需要学习 | | 容器提交 | 临时修改、快速保存状态 | 操作简单、快速 | 难以追溯变更、不易自动化 | > 💡 最佳实践:生产环境**强烈推荐使用 Dockerfile** 方式,便于实现基础设施即代码(IaC)和 CI/CD 流程[^1][^2]。 --- ### 镜像管理辅助命令 1. **查看镜像列表**: ```bash docker images ``` 2. **导出/导入镜像**: ```bash # 导出镜像 docker save -o my-nginx.tar my-nginx-image:latest # 导入镜像 docker load -i my-nginx.tar ``` 3. **删除镜像**: ```bash docker rmi my-nginx-image:latest ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

THMAIL

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值