Java学习之Docker镜像讲解

本文详细介绍了Docker镜像的概念,包括镜像的定义、Docker镜像加载原理(基于UnionFS的分层结构)、以及如何理解和利用分层特性。此外,还通过实例演示了如何创建和提交自定义镜像,以及Docker存储引擎的工作原理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、镜像是什么

镜像是一种轻量级、可执行的独立软件包,用来打包软件运行环境和基于运行环境开发的软件,它包含运行某个软件所需的所有内容,包括代码、运行时、库、环境变量和配置文件。

二、Docker镜像加载原理

UnionFS(联合文件系统)

联合文件系统(UnionFS)是一种分层、轻量级并且高性能的文件系统,它支持对文件系统的修改作为一次提交来一层层的叠加,同时可以将不同目录挂载到同一个虚拟文件系统下(unite several directories into a single virtual filesystem)。联合文件系统是 Docker 镜像的基础。镜像可以通过分层来进行继承,基于基础镜像(没有父镜像),可以制作各种具体的应用镜像。
特性:一次同时加载多个文件系统,但从外面看起来,只能看到一个文件系统,联合加载会把各层文件系统叠加起来,这样最终的文件系统会包含所有底层的文件和目录

Docker镜像加载原理

docker的镜像实际上由一层一层的文件系统组成,这种层级的文件系统UnionFS。bootfs(boot file system)主要包含bootloader和kernel,bootloader主要是引导加载kernel,Linux刚启动时会加载bootfs文件系统,在Docker镜像的最底层是bootfs。这一层与我们典型的Linux/Unix系统是一样的,包含boot加载器和内核。当boot加载完成之后整个内核就都在内存中了,此时内存的使用权已由bootfs转交给内核,此时系统也会卸载bootfs。
rootfs (root file system),在bootfs之上。包含的就是典型Linux 系统中的/dev,/proc,/bin,/etc 等标准目录和文件。rootfs就是各种不同的操作系统发行版,比如Ubuntu,centos等等。
在这里插入图片描述
平时我们安装进虚拟机的CentOs都是好几个G,为什么Docker这里才200M ?

[root@master /]$ docker images
REPOSITORY      TAG       IMAGE ID       CREATED       SIZE
tomcat          latest    93b7b04e85ea   3 days ago    448MB
nginx           latest    8aea65d81da2   8 weeks ago   192MB
centos          latest    e6a0117ec169   2 years ago   272MB
elasticsearch   latest    5acf0e8da90b   5 years ago   486MB

对于一个精简的0S,rootfs 可以很小,只需要包含最基本的命令,工具和程序库就可以了,因为底层直接用Host的kernel,自己只需要提供rootfs就可以了。由此可见对于不同的linux发行版,bootfs基本是一致的,rootfs会有差别,因此不同的发行版可以公用
bootfs。

三、分层理解

分层的镜像

[root@master /]$ docker pull elasticsearch
Using default tag: latest
latest: Pulling from library/elasticsearch
[DEPRECATION NOTICE] Docker Image Format v1, and Docker Image manifest version 2, schema 1 support will be removed in an upcoming release. Suggest the author of docker.io/library/elasticsearch:latest to upgrade the image to the OCI Format, or Docker Image manifest v2, schema 2. More information at https://docs.docker.com/go/deprecated-image-specs/
05d1a5232b46: Pull complete 
5cee356eda6b: Pull complete 
89d3385f0fd3: Pull complete 
65dd87f6620b: Pull complete 

所有的 Docker 镜像都起始于一个基础镜像层,当进行修改或增加新的内容时,就会在当前镜像层之上,创建新的镜像层。举一个简单的例子,假如基于 Ubuntu Linux16.04创建一个新的镜像,这就是新镜像的第一层;如果在该镜像中添加 Python包就会在基础镜像层之上创建第二个镜像层;如果继续添加一个安全补丁,就会创建第三个镜像层。该镜像当前已经包含3个镜像层,如下图所示(这只是一个用于演示的很简单的例子)。
在这里插入图片描述
在添加额外的镜像层的同时,镜像始终保持是当前所有镜像的组合,理解这一点非常重要。下图中举了一个简单的例子,每个镜像层包含3个文件,而镜像包含了来自两个镜像层的6个文件。
在这里插入图片描述
上图中的镜像层跟之前图中的略有区别,主要目的是便于展示文件。
下图中展示了一个稍微复杂的三层镜像,在外部看来整个镜像只有6个文件,这是因为最上层中的文件7是文件5的一个更新版本。
在这里插入图片描述
这种情况下,上层镜像层中的文件覆盖了底层镜像层中的文件。这样就使得文件的更新版本作为一个新镜像层添加到镜像当中。
Docker 通过存储引擎(新版本采用快照机制)的方式来实现镜像层堆栈,并保证多镜像层对外展示为统一的文件系统。
Linux 上可用的存储引擎有 AUFS、Overlav2、Device Mapper、Btrfs 以及ZFS。顾名思义,每种存储引擎都基于 Linux 中对应的文件系统或者块设备技术,并且每种存储引警都有其独有的性能特点。
Docker 在 Windows 上仅支持 windowsfilter 一种存储引擎,该引擎基于 NTFS 文件系统之上实现了分层和 CoW[1]。
下图展示了与系统显示相同的三层镜像。所有镜像层堆叠并合并,对外提供统一的视图。
在这里插入图片描述

特点

Docker镜像都是只读的,当容器启动时,一个新的可写层被加载到镜像的顶部!
这一层就是我们通常说的容器层,容器之下的都叫镜像层!
在这里插入图片描述
那么如何提交一个自己的镜像呢?

commit镜像


# 提交一个新的副本
docker commit

# 命令和git原理类似
docker commit -m="提交的描述信息" -a="作者" 容器id 目标镜像名:[TAG]
实战测试

1、启动一个默认的tomcat
2、发现这个默认的tomcat是没有webapps应用的,这是镜像的原因,官方的镜像默认webapps下面是没有文件的!
3、那么我们自己拷贝进去基本的文件

[root@master /]$ docker run -d -p 3366:8080 --name tomcat02 tomcat
c97ed91f1107ee8ef38c9138f9ba20ed2b8fea4a140c2fd5b189b94405bc7b4a
[root@master /]$ docker ps
CONTAINER ID   IMAGE     COMMAND             CREATED         STATUS         PORTS                                       NAMES
c97ed91f1107   tomcat    "catalina.sh run"   8 seconds ago   Up 8 seconds   0.0.0.0:3366->8080/tcp, :::3366->8080/tcp   tomcat02
[root@master /]$ docker exec -it c97ed91f1107 /bin/bash
root@c97ed91f1107:/usr/local/tomcat$ 
root@c97ed91f1107:/usr/local/tomcat$ 
root@c97ed91f1107:/usr/local/tomcat$ cp -r webapps.dist/* webapps
root@c97ed91f1107:/usr/local/tomcat$ ls
bin              lib             NOTICE         temp
BUILDING.txt     LICENSE         README.md      webapps
conf             logs            RELEASE-NOTES  webapps.dist
CONTRIBUTING.md  native-jni-lib  RUNNING.txt    work
root@c97ed91f1107:/usr/local/tomcat$ cd webapps
root@c97ed91f1107:/usr/local/tomcat/webapps$ ls
docs  examples  host-manager  manager  ROOT

4、将我们操作过的容器通过commit提交为一个镜像,我们以后就可以使用我们修改过的镜像了,这就是我们自己修改过的一个镜像。

[root@master /]$ docker commit -m="添加了基本信息" -a="makabaka" c97ed91f1107 tomcat02:1.0
sha256:ca350bb930750ada6028e576d1ec9827f434a42d74b896f8efc37d54da9fce68
[root@master /]$ docker images
REPOSITORY      TAG       IMAGE ID       CREATED         SIZE
tomcat02        1.0       ca350bb93075   6 seconds ago   452MB
tomcat          latest    93b7b04e85ea   3 days ago      448MB
nginx           latest    8aea65d81da2   8 weeks ago     192MB
centos          latest    e6a0117ec169   2 years ago     272MB
elasticsearch   latest    5acf0e8da90b   5 years ago     486MB
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值