Docker 学习
- Docker概述
- Docker安装
- Docker命令
- 镜像命令
- 容器命令
- 操作命令
- ...
- Docker镜像
- 容器数据卷
- DockerFile
- Docker网络原理
- IDEA整合Docker
- Docker Compose
- Docker Swarm
- CI/CD Jenkins
Docker概述
Docker为什么出现?
一款产品:开发 --> 上线 两套环境!应用环境,应用配置!
开发 --> 运维 问题:我在我的电脑上可以运行!版本更新,导致服务不可用!对于运维来说,考验就十分大!
环境配置是十分的麻烦,每一个机器都要部署环境(集群Redis、ES、Hadoop......)!费时费力。
发布一个项目(jar + (Redis MySQL jdk ES)),项目能不能都带上环境安装打包!
之前在服务器配置一个应用环境 Redis MySQL jdk ES Hadoop,配置十分麻烦,不能够跨平台。
Windows,最后发布到Linux!
传统:开发 jar,运维来做!
现在:开发打包部署上线,一套流程做完!
Java --> apk --> 发布(应用商店) --> 张三使用apk --> 安装即可用!
Java --> jar(环境) --> 打包项目带上环境(镜像) --> (Docker仓库:商店) --> 下载我们发布的镜像 --> 直接运行即可!
Docker给以上的问题,提出了解决方案!
Docker的思想就来自于集装箱!
JRE --> 多个应用(端口冲突) --> 原来都是交叉的!
隔离:Docker核心思想:打包装箱!每个箱子是互相隔离的。
水果 生化武器
Docker 通过隔离机制,可以将服务器利用到极致!
本质:所有的技术都是因为出现了一些问题,我们需要去解决,才去学习!
Docker的历史
2010年,几个搞IT的年轻人,就在美国成立了一家dotCloud
做一些pass
的云计算服务!LXC 有关的容器技术!
他们将自己的技术(容器化技术)命名 就是 Docker!
Docker 刚刚诞生的时候,没有引起行业的注意!dotCloud,就活不下去!
开源
开放源代码!
2013年,Docker开源!
Docker越来越多的人发现了Docker的优点!火了,Docker 每个月都会更新一个版本!
2014年4月9日,Docker1.0发布!
Docker为什么这么火?十分的轻巧!
在容器技术出来之前,我们都是使用虚拟机技术!
虚拟机:在Windows中装一个VMware,通过这个软件我们可以虚拟出来一台或者多台电脑!十分笨重!
虚拟机也是属于虚拟化技术,Docker 容器技术,也是一种 虚拟化技术!
vm : linux centos原生镜像(一个电脑!) 隔离,需要开启多个虚拟机! 几个G 几分钟 docker : 隔离,镜像(最核心的环境 4m + jdk + mysql)十分的小巧,运行镜像就可以了!小巧! 几个M KB 秒级启动!
到现在,所有开发人员都必须要会Docker!
聊聊Docker
Docker基于Go语言开发的!开源项目!
文档地址:https://docs.docker.com/ Docker的文档是超级详细的!
Docker能干嘛
之前的虚拟机技术!
虚拟机技术缺点:
- 资源占用十分多
- 冗余步骤多
- 启动很慢!
容器化技术
容器化技术不是模拟的一个完整的操作系统
比较Docker 和 虚拟机技术的不同:
- 传统虚拟机,虚拟出一套硬件,运行一个完整的操作系统,然后在这个系统上安装和运行软件
- 容器内的应用直接运行在宿主机的内核,容器是没有自己的内核的,也没有虚拟我们的硬件,所以就轻便了
- 每个容器间是互相隔离的,每个容器内都有一个属于自己的文件系统,互不影响。
DevOps(开发、运维)
应用更快速的交付和部署
传统:一堆帮助文档,安装程序
Docker:打包镜像,发布测试,一键运行
更便捷的升级和扩缩容
使用了Docker之后,我们部署应用就和搭积木一样!
项目打包为一个镜像,扩展 服务器A!服务器B
更简单的系统运维
在容器化之后,我们的开发、测试环境都是高度一致的。
更高效的计算资源利用
Docker 是内核级别的虚拟化,可以在物理机上可以运行很多的容器实例!服务器的性能可以被压榨到极致。
Docker 安装
Docker的基本组成
镜像(image):
docker镜像就像一个模板,可以通过这个模板来创建服务,tomcat镜像 ==> run ==> tomcat01容器(提供服务),通过这个镜像可以创建多个容器(最终服务运行或者项目运行就是在容器中的)。
容器(container):
Docker利用容器技术,独立运行一个或者一组应用,通过镜像来创建。
启动,停止,删除,基本命令!
目前可以把这个容器理解为就是一个简易的linux系统
仓库(repository):
仓库就是存放镜像的地方!
仓库分为公有仓库和私有仓库!
Docker Hub(默认是国外的)
阿里云...都有容器服务器(配置镜像加速!)
安装Docker
环境准备
- 需要会一点点的Linux基础
- CentOS 7
- 我们使用Xshell连接远程服务器进行操作!
环境查看
# 系统内核是 3.10 以上的 # 系统版本
安装
帮助文档:
# 1、卸载旧的版本 sudo apt-get remove docker docker-engine docker.io containerd runc # 2、需要的安装包 yum install -y yum-utils # 3、设置镜像的仓库 yum-config-manager --add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo # 阿里云 # 更新yum软件包索引 yum makecache fast # 4、安装docker docker-ce 社区版 ee 企业版 yum install docker-ce docker-ce-cli containerd.io # 5、启动docker systemctl start docker # 6、使用docker version 查看是否安装成功
# 7、hello-world docker run hello-world
# 8、查看一下下载的这个 hello-world 镜像 [root@ls-Cwj2oH9C /]# docker images REPOSITORY TAG IMAGE ID CREATED SIZE hello-world latest feb5d9fea6a5 9 months ago 13.3kB
了解:卸载docker
# 1、卸载依赖 yum remove docker-ce docker-ce-cli containerd.io # 2、删除资源 rm -rf /var/lib/docker # /var/lib/docker docker的默认工作路径!
阿里云镜像加速
- 登录阿里云找到容器服务
- 找到镜像加速的地址
- 配置使用
sudo mkdir -p /etc/docker sudo tee /etc/docker/daemon.json <<-'EOF' { "registry-mirrors": ["https://veotnqhz.mirror.aliyuncs.com"] } EOF sudo systemctl daemon-reload sudo systemctl restart docker
回顾HelloWorld流程
底层原理
Docker是怎么工作的?
Docker是一个Client - Server 结构的系统,Docker的守护进程运行在主机上。通过Socket从客户端访问!
DockerServer接收到 Docker - Client 的指令,就会执行这个命令!
Docker 为什么比 VM 快?
- Docker有着比虚拟机更少的抽象层
- Docker利用的是宿主机的内核,vm需要的是Guest OS。
所以说,新建一个容器的时候,docker不需要像虚拟机一样重新加载一个操作系统内核,避免引导操作。虚拟机是加载Guest OS,分钟级别的,而docker是利用宿主机的操作系统,省略了这个复杂的过程,秒级!
之后学习完毕所有的命令,再回过头来看这段理论,就会很清晰!
Docker的常用命令
帮助命令
docker version # 显示docker的版本信息 docker info # 显示docker的系统信息,包括镜像和容器的数量 docker 命令 --help # 帮助命令
帮助文档的地址:https://docs.docker.com/engine/reference/commandline/docker/
镜像命令
docker images 查看所有本地的主机上的镜像
[root@ls-Cwj2oH9C /]# docker images REPOSITORY TAG IMAGE ID CREATED SIZE hello-world latest feb5d9fea6a5 9 months ago 13.3kB # 解释 REPOSITORY 镜像的仓库源 TAG 镜像的标签 IMAGE ID 镜像的id CREATED 镜像的创建时间 SIZE 镜像的大小 # 可选项 -a,--all # 列出所有镜像 -q,--quiet # 只显示镜像的id
docker search 搜索镜像
[root@ls-Cwj2oH9C /]# docker search mysql NAME DESCRIPTION STARS OFFICIAL AUTOMATED mysql MySQL is a widely used, open-source relation… 12821 [OK] mariadb MariaDB Server is a high performing open sou… 4918 [OK] percona Percona Server is a fork of the MySQL relati… 580 [OK] phpmyadmin phpMyAdmin - A web interface for MySQL and M… 564 [OK] # 可选项,通过搜索来过滤 --filter=STARS=3000 # 搜索出来的镜像就是STARS大于3000的 [root@ls-Cwj2oH9C ~]# docker search mysql --filter=STARS=3000 NAME DESCRIPTION STARS OFFICIAL AUTOMATED mysql MySQL is a widely used, open-source relation… 12821 [OK] mariadb MariaDB Server is a high performing open sou… 4918 [OK] [root@ls-Cwj2oH9C ~]# docker search mysql --filter=STARS=5000 NAME DESCRIPTION STARS OFFICIAL AUTOMATED mysql MySQL is a widely used, open-source relation… 12821 [OK]
docker pull 下载镜像
# 下载镜像 docker pull 镜像名[:tag] [root@ls-Cwj2oH9C /]# docker pull mysql Using default tag: latest # 如果不写 tag,默认就是latest latest: Pulling from library/mysql 72a69066d2fe: Pull complete # 分层下载,docker image的核心 联合文件系统 93619dbc5b36: Pull complete 99da31dd6142: Pull complete 626033c43d70: Pull complete 37d5d7efb64e: Pull complete ac563158d721: Pull complete d2ba16033dad: Pull complete 688ba7d5c01a: Pull complete 00e060b6d11d: Pull complete 1c04857f594f: Pull complete 4d7cfa90e6ea: Pull complete e0431212d27d: Pull complete Digest: sha256:e9027fe4d91c0153429607251656806cc784e914937271037f7738bd5b8e7709 # 签名 Status: Downloaded newer image for mysql:latest docker.io/library/mysql:latest # 真实地址 # 等价于它 docker pull mysql docker pull docker.io/library/mysql:latest # 指定版本下载 [root@ls-Cwj2oH9C /]# docker pull mysql:5.7 5.7: Pulling from library/mysql 72a69066d2fe: Already exists 93619dbc5b36: Already exists 99da31dd6142: Already exists 626033c43d70: Already exists 37d5d7efb64e: Already exists ac563158d721: Already exists d2ba16033dad: Already exists 0ceb82207cd7: Pull complete 37f2405cae96: Pull complete e2482e017e53: Pull complete 70deed891d42: Pull complete Digest: sha256:f2ad209efe9c67104167fc609cca6973c8422939491c9345270175a300419f94 Status: Downloaded newer image for mysql:5.7 docker.io/library/mysql:5.7
docker rmi 删除镜像!
[root@ls-Cwj2oH9C /]# docker rmi -f 镜像id # 删除指定的镜像 [root@ls-Cwj2oH9C /]# docker rmi -f 镜像id 镜像id 镜像id 镜像id # 删除多个镜像 [root@ls-Cwj2oH9C /]# docker rmi -f $(docker images -aq) # 删除全部的镜像
容器命令
说明:我们有了镜像才可以创建容器,linux,下载一个 centos 镜像来测试学习
docker pull centos
新建容器并启动
docker run [可选参数] image # 参数说明 --name="Name" 容器名字 tomcat01 tomcat02,用来区分容器 -d 后台方式运行 -it 使用交互方式运行,进入容器查看内容 -p 指定容器的端口 -p 8080:8080 -p ip:主机端口:容器端口 -p 主机端口:容器端口 (常用) -p 容器端口 容器端口 -p 随机指定端口 # 测试,启动并进入容器 [root@ls-Cwj2oH9C ~]# docker run -it centos /bin/bash [root@26c5b40b2e60 /]# ls # 查看容器内的centos,基础版本,很多命令都是不完善的! bin etc lib lost+found mnt proc run srv tmp var dev home lib64 media opt root sbin sys usr # 从容器中退回主机 [root@c103dbc1d4d4 /]# exit exit [root@ls-Cwj2oH9C ~]# cd / [root@ls-Cwj2oH9C /]# ls bin dev home lib64 media opt root sbin sys usr boot etc lib lost+found mnt proc run srv tmp var
列出所有的运行的容器
# docker ps 命令 # 列出当前正在运行的容器 -a # 列出当前正在运行的容器+带出历史运行过的容器 -n=? # 显示最近创建的容器 [root@ls-Cwj2oH9C /]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES [root@ls-Cwj2oH9C /]# docker ps -a CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES c103dbc1d4d4 centos "/bin/bash" 57 seconds ago Exited (0) 50 seconds ago busy_ramanujan 26c5b40b2e60 centos "/bin/bash" 11 minutes ago Exited (0) About a minute ago thirsty_brahmagupta 1aba774b8be5 feb5d9fea6a5 "/hello" 4 hours ago Exited (0) 4 hours ago quizzical_turing [root@ls-Cwj2oH9C /]# docker ps -aq c103dbc1d4d4 26c5b40b2e60 1aba774b8be5
退出容器
exit # 直接容器停止并退出 Ctrl + P + Q # 容器不停止退出
删除容器
docker rm 容器id # 删除指定的容器,不能删除正在运行的容器,如果要强制删除 rm -f docker rm -f $(docker ps -aq) # 删除所有的容器 docker ps -a -q|xargs docker rm # 删除所有的容器
启动和停止容器的操作
docker start 容器id # 启动容器 docker restart 容器id # 重启容器 docker stop 容器id # 停止当前正在运行的容器 docker kill 容器id # 强制停止当前容器
常用其他命令
后台启动容器
# 命令 docker run -d 镜像名! [root@ls-Cwj2oH9C /]# docker run -d centos d3ebfe17b9c02d91f4f45bc9405b6713b7098e9c2f1229cd6523b46f85531c03 # 问题docker ps,发现 centos 停止了 # 常见的坑:docker 容器使用后台运行,就必须要有一个前台进程,docker发现没有应用,就会自动停止 # nginx,容器启动后,发现自己没有提供服务,就会立刻停止,就是没有程序了
查看日志
docker logs -f -t --tail 10 容器 没有日志 # 自己编写一段shell脚本 [root@ls-Cwj2oH9C /]# docker run -d centos /bin/sh -c "while true;do echo kuangshen;sleep 1;done" be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de # [root@ls-Cwj2oH9C /]# docker ps CONTAINER ID IMAGE be09e0fd41c7 centos d2ae011f21b8 centos # 显示日志 -tf # 显示日志 --tail number # 要显示的日志条数 [root@ls-Cwj2oH9C /]# docker logs -tf --tail 10 be09e0fd41c7
查看容器中进程信息 ps
# 命令 docker top 容器id [root@ls-Cwj2oH9C /]# docker top be09e0fd41c7 UID PID PPID C STIME root 23558 23538 0 15:53 root 25465 23558 0 16:09
查看镜像的元数据
# 命令 docker inspect 容器id # 测试 [root@ls-Cwj2oH9C /]# docker inspect be09e0fd41c7 [ { "Id": "be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de", "Created": "2022-07-05T07:53:54.627754682Z", "Path": "/bin/sh", "Args": [ "-c", "while true;do echo kuangshen;sleep 1;done" ], "State": { "Status": "running", "Running": true, "Paused": false, "Restarting": false, "OOMKilled": false, "Dead": false, "Pid": 23558, "ExitCode": 0, "Error": "", "StartedAt": "2022-07-05T07:53:54.913270812Z", "FinishedAt": "0001-01-01T00:00:00Z" }, "Image": "sha256:5d0da3dc976460b72c77d94c8a1ad043720b0416bfc16c52c45d4847e53fadb6", "ResolvConfPath": "/var/lib/docker/containers/be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de/resolv.conf", "HostnamePath": "/var/lib/docker/containers/be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de/hostname", "HostsPath": "/var/lib/docker/containers/be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de/hosts", "LogPath": "/var/lib/docker/containers/be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de/be09e0fd41c70e5d114b5c2534acba6ae97a3a12e1b4d0d603077b83c76697de-json.log", "Name": "/vigilant_tu", "RestartCount": 0, "Driver": "overlay2", "Platform": "linux", "MountLabel": "", "ProcessLabel": "", "AppArmorProfile": "", "ExecIDs": null, "HostConfig": { "Binds": null, "ContainerIDFile": "", "LogConfig": { "Type": "json-file", "Config": {} }, "NetworkMode": "default", "PortBindings": {}, "RestartPolicy": { "Name": "no", "MaximumRetryCount": 0 }, "AutoRemove": false, "VolumeDriver": "", "VolumesFrom": null, "CapAdd": null, "CapDrop": null, "CgroupnsMode": "host", "Dns": [], "DnsOptions": [], "DnsSearch": [], "ExtraHosts": null, "GroupAdd": null, "IpcMode": "private", "Cgroup": "", "Links": null, "OomScoreAdj": 0, "PidMode": "", "Privileged": false, "PublishAllPorts": false, "ReadonlyRootfs": false, "SecurityOpt": null, "UTSMode": "", "UsernsMode": "", "ShmSize": 67108864, "Runtime": "runc", "ConsoleSize": [ 0, 0 ], "Isolation": "", "CpuShares": 0, "Memory": 0, "NanoCpus": 0, "CgroupParent": "", "BlkioWeight": 0, "BlkioWeightDevice": [], "BlkioDeviceReadBps": null, "BlkioDeviceWriteBps": null, "BlkioDeviceReadIOps": null, "BlkioDeviceWriteIOps": null, "CpuPeriod": 0, "CpuQuota": 0, "CpuRealtimePeriod": 0, "CpuRealtimeRuntime": 0, "CpusetCpus": "", "CpusetMems": "", "Devices": [], "DeviceCgroupRules": null, "DeviceRequests": null, "KernelMemory": 0, "KernelMemoryTCP": 0, "MemoryReservation": 0, "MemorySwap": 0, "MemorySwappiness": null, "OomKillDisable": false, "PidsLimit": null, "Ulimits": null, "CpuCount": 0, "CpuPercent": 0, "IOMaximumIOps": 0, "IOMaximumBandwidth": 0, "MaskedPaths": [ "/proc/asound", "/proc/acpi", "/proc/kcore", "/proc/keys", "/proc/latency_stats", "/proc/timer_list", "/proc/timer_stats", "/proc/sched_debug", "/proc/scsi", "/sys/firmware" ], "ReadonlyPaths": [ "/proc/bus", "/proc/fs", "/proc/irq", "/proc/sys", "/proc/sysrq-trigger" ] }, "GraphDriver": { "Data": { "LowerDir": "/var/lib/docker/overlay2/1a375808cf6a2eff0ce527432ddf9fd9b2c5519b1a71b9c6a5931104f24e55ed-init/diff:/var/lib/docker/overlay2/652792c86aac30ea5e1860169ae46b72da12c74efde67ea5aaba90a396e5989f/diff", "MergedDir": "/var/lib/docker/overlay2/1a375808cf6a2eff0ce527432ddf9fd9b2c5519b1a71b9c6a5931104f24e55ed/merged", "UpperDir": "/var/lib/docker/overlay2/1a375808cf6a2eff0ce527432ddf9fd9b2c5519b1a71b9c6a5931104f24e55ed/diff", "WorkDir": "/var/lib/docker/overlay2/1a375808cf6a2eff0ce527432ddf9fd9b2c5519b1a71b9c6a5931104f24e55ed/work" }, "Name": "overlay2" }, "Mounts": [], "Config": { "Hostname": "be09e0fd41c7", "Domainname": "", "User": "", "AttachStdin": false, "AttachStdout": false, "AttachStderr": false, "Tty": false, "OpenStdin": false, "StdinOnce": false, "Env": [ "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin" ], "Cmd": [ "/bin/sh", "-c", "while true;do echo kuangshen;sleep 1;done" ], "Image": "centos", "Volumes": null, "WorkingDir": "", "Entrypoint": null, "OnBuild": null, "Labels": { "org.label-schema.build-date": "20210915", "org.label-schema.license": "GPLv2", "org.label-schema.name": "CentOS Base Image", "org.label-schema.schema-version": "1.0", "org.label-schema.vendor": "CentOS" } }, "NetworkSettings": { "Bridge": "", "SandboxID": "4b2ab1f2e949baa34edcfd36e72e6ed4a80c5f5381f6183b30ee6d9e73e39e44", "HairpinMode": false, "LinkLocalIPv6Address": "", "LinkLocalIPv6PrefixLen": 0, "Ports": {}, "SandboxKey": "/var/run/docker/netns/4b2ab1f2e949", "SecondaryIPAddresses": null, "SecondaryIPv6Addresses": null, "EndpointID": "83295610dca0bd58cf285f6e7238eaae06749bf30433a75ee9d126dd6f92c25f", "Gateway": "172.17.0.1", "GlobalIPv6Address": "", "GlobalIPv6PrefixLen": 0, "IPAddress": "172.17.0.3", "IPPrefixLen": 16, "IPv6Gateway": "", "MacAddress": "02:42:ac:11:00:03", "Networks": { "bridge": { "IPAMConfig": null, "Links": null, "Aliases": null, "NetworkID": "4181a198dd37ad281366217d8321f1ca65fd69f9139e73c0a7fc460afc58fa28", "EndpointID": "83295610dca0bd58cf285f6e7238eaae06749bf30433a75ee9d126dd6f92c25f", "Gateway": "172.17.0.1", "IPAddress": "172.17.0.3", "IPPrefixLen": 16, "IPv6Gateway": "", "GlobalIPv6Address": "", "GlobalIPv6PrefixLen": 0, "MacAddress": "02:42:ac:11:00:03", "DriverOpts": null } } } } ]
进入当前正在运行的容器
# 我们通常容器都是使用后台方式运行的,需要进入容器,修改一些配置 # 命令 docker exec -it 容器id bashShell # 测试 [root@ls-Cwj2oH9C /]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES be09e0fd41c7 centos "/bin/sh -c 'while t…" 32 minutes ago Up 32 minutes vigilant_tu d2ae011f21b8 centos "/bin/bash" 45 minutes ago Up 45 minutes mystifying_tereshkova [root@ls-Cwj2oH9C /]# docker exec -it be09e0fd41c7 /bin/bash [root@be09e0fd41c7 /]# ls bin etc lib lost+found mnt proc run srv tmp var dev home lib64 media opt root sbin sys usr [root@be09e0fd41c7 /]# ps -ef UID PID PPID C STIME TTY TIME CMD root 1 0 0 07:53 ? 00:00:01 /bin/sh -c while true;do echo kuangshen;sleep 1;don root 4191 0 0 09:03 pts/0 00:00:00 /bin/bash root 4214 1 0 09:03 ? 00:00:00 /usr/bin/coreutils --coreutils-prog-shebang=sleep / root 4215 4191 0 09:03 pts/0 00:00:00 ps -ef # 方式二 docker attach 容器id # 测试 [root@ls-Cwj2oH9C /]# docker attach be09e0fd41c7 正在执行当前的代码... # docker exec # 进入容器后开启一个新的终端,可以在里面操作(常用) # docker attach # 进入容器正在执行的终端,不会启动新的进程!
从容器内拷贝文件到主机上
docker cp 容器id:容器内路径 目的主机路径 # 查看当前主机目录下 [root@ls-Cwj2oH9C home]# ls kuangshen.java [root@ls-Cwj2oH9C home]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES 17c1f83eeda6 centos "/bin/bash" 32 seconds ago Up 31 seconds relaxed_spence be09e0fd41c7 centos "/bin/sh -c 'while t…" 2 hours ago Up 2 hours vigilant_tu d2ae011f21b8 centos "/bin/bash" 2 hours ago Up 2 hours mystifying_tereshkova # 进入docker容器内部 [root@ls-Cwj2oH9C home]# docker attach 17c1f83eeda6 [root@17c1f83eeda6 /]# cd /home [root@17c1f83eeda6 home]# ls # 在容器内新建一个文件 [root@17c1f83eeda6 home]# touch test.java [root@17c1f83eeda6 home]# exit exit [root@ls-Cwj2oH9C home]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES be09e0fd41c7 centos "/bin/sh -c 'while t…" 2 hours ago Up 2 hours vigilant_tu d2ae011f21b8 centos "/bin/bash" 2 hours ago Up 2 hours mystifying_tereshkova [root@ls-Cwj2oH9C home]# docker ps -a CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES 17c1f83eeda6 centos "/bin/bash" 2 minutes ago Exited (0) 7 seconds ago relaxed_spence be09e0fd41c7 centos "/bin/sh -c 'while t…" 2 hours ago Up 2 hours vigilant_tu d2ae011f21b8 centos "/bin/bash" 2 hours ago Up 2 hours mystifying_tereshkova d3ebfe17b9c0 centos "/bin/bash" 3 hours ago Exited (0) 3 hours ago gifted_joliot # 将这个文件拷贝出来到主机上 [root@ls-Cwj2oH9C home]# docker cp 17c1f83eeda6:/home/test.java /home [root@ls-Cwj2oH9C home]# ls kuangshen.java test.java [root@ls-Cwj2oH9C home]# # 拷贝是一个手动过程,未来我们使用 -v 卷的技术,可以实现,自动同步