我曾在之前的一篇文章中介绍了有关docker的一些基本的操作(Docker学习笔记),最近一直在玩Docker,一种应用程序容器和Linux的虚拟技术。它太酷了,创建Docker镜像和容器只需要几分钟。所有的工作都是开箱即用的。
学习归学习,但是之后我希望能保存下我的工作。在Docker的save和export命令之间,我凌乱了~~~
Docker是如何工作的(简单说明)
Docker是基于镜像的。镜像类似于已经包含了文件、配置和安装好的程序的虚拟机镜像。同样的,你可以像启动虚拟机一样启动多个镜像实例。运行中的镜像称为容器。你可以修改容器(比如删除一个文件),但这些修改不会影响到镜像。不过,你使用docker commit <container_id> <image_name>命令可以把一个正在运行的容器变成一个新的镜像。
举个例子:
# 拉取一个ubuntu的镜像 sudo docker pull ubuntu:14.04 # 查看本地已经有哪些镜像 ,我们可以看到ubuntu docker images # 现在让我们来修改下ubuntu镜像的容器 # 这次,我们创建一个文件夹 sudo docker run ubuntu:14.04 mkdir /home/test # 让我们再看看我们有哪些镜像了。 # 注意每条命令执行后容器都会停止 # 可以看到有一个ubuntu容器 docker ps -a # 现在,可以提交修改了。 # 提交后会看到一个新的镜像ubuntuv1 # <CONTAINER_ID> 是刚刚修改容器后得到的ID sudo docker commit <CONTAINER_ID> ubuntuv1:14.04 # 再看看我们有哪些镜像。 # 我们现在同时有ubuntu和ubuntuv1镜像了。 docker images # 我们执行以下命令,看看这两个镜像有什么不同 sudo docker run ubuntu:14.04 [ -d /home/test ] && echo 'Directory found' || echo 'Directory not found' sudo docker run ubuntuv1:14.04 [ -d /home/test ] && echo 'Directory found' || echo 'Directory not found'
现在,我们有两个不同的镜像了(ubuntu和ubuntuv1),还有一个通过修改ubuntu容器得来的容器(多了一个/home/test文件夹)。下面来看看,是如何持久化这些修改的。
导出(Export)export命令用于持久化容器(不是镜像)。所以,我们就需要通过以下方法得到容器ID,接着执行导出:
docker ps -a sudo docker export <CONTAINER_ID> > /home/export.tar
最后的结果是一个tar文件(比使用save命令稍微小些,基本上差不多)。
保存(Save)save命令用于持久化镜像(不是容器)。所以,我们就需要通过以下方法得到镜像名称,接着执行保存:
docker images sudo docker save ubuntuv1:14.04 > /home/save.tar //或者使用其ID
最后的结果是一个tar文件(比使用export命令稍微大些)。
它们之间的不同现在我们创建了两个tar文件,让我们来看看它们是什么。首先做一下小清理——把所有的容器和镜像都删除:
# 查看所有的容器 sudo docker ps -a # 删除它们 sudo docker stop <CONTAINER_ID> sudo docker rm <CONTAINER_ID> //在此也可以一次性全部删除 sudo docker stop $(docker ps -a -q) ; sudo docker rm $(docker ps -a -q) # 查看所有的镜像 docker images # 删除它们 sudo docker rmi ubuntu //或者使用其ID sudo docker rmi ubuntuv1 //或者使用其ID //在此也可以一次性全部删除 sudo docker rmi $(docker images -q)
现在开始导入刚刚导出的容器:
# 导入export.tar文件 cat /home/export.tar | sudo docker import - ubuntu_export:14.04 # 查看镜像 docker images # 检查是否导入成功,就是启动一个新容器,检查里面是否存在/home/test目录(是存在的) sudo docker run ubuntu_export:14.04 [ -d /home/test ] && echo 'Directory found' || echo 'Directory not found'
使用类似的步骤导入镜像:
# 导入save.tar文件 docker load < /home/save.tar # 查看镜像 docker images # 检查是否导入成功,就是启动一个新容器,检查里面是否存在/home/test目录(是存在的) sudo docker run ubuntuv1:14.04 [ -d /home/test ] && echo 'Directory found' || echo 'Directory not found'
那它们之间到底存在什么不同呢?我们发现导出后的版本会比原来的版本稍微小一些。那是因为导出后,会丢失历史和元数据。执行下面的命令就知道了:
# 显示镜像的所有层(layer) sudo docker images --tree
执行命令,显示下面的内容。正你看到的,导出后再导入(exported-imported)的镜像会丢失所有的历史,而保存后再加载(saveed-loaded)的镜像没有丢失历史和层(layer)。这意味着使用导出后再导入的方式,你将无法回滚到之前的层(layer),同时,使用保存后再加载的方式持久化整个镜像,就可以做到层回滚(可以执行docker tag <LAYER ID> <IMAGE NAME>来回滚之前的层)。
vagrant@ubuntu-13:~$ sudo docker images --tree ├─f502877df6a1 Virtual Size: 2.489 MB Tags: busybox-1-export:latest └─511136ea3c5a Virtual Size: 0 B └─bf747efa0e2f Virtual Size: 0 B └─48e5f45168b9 Virtual Size: 2.489 MB └─769b9341d937 Virtual Size: 2.489 MB └─227516d93162 Virtual Size: 2.489 MB Tags: busybox-1:latest
到此完毕~~
当你工作做完了之后,重启PC时,再执行docker 相关的命令时发现会出现这样的错误,切无法执行相关的操作了:
Get http:///var/run/docker.sock/v1.12/images/json: dial unix /var/run/docker.sock: no such file or directory
在网上也有以下解决的方法,例如:
1,你可以看以下你的用户是否在docker用户组, /var/run/docker.sock的访问权限只root和docker用户组, 所以你需要把自己的用户添加到docker组, 命令: sudo gpasswd -a ${uesrname} docker 2,先运行 sudo docker -d -H unix:///var/run/docker.sock 启动服务 3,用env看你的环境变量设置的有没有问题,注意是否用了sudo。 $ export DOCKER_TLS_VERIFY=1 $ export DOCKER_HOST=tcp://192.168.59.103:2376 $ export DOCKER_CERT_PATH=/Users/myusername/.boot2docker/certs/boot2docker-vm
上面的解决方法我也曾试过,不但没有解决问题,而且还会出现其他的错误。
正确的解决方法是:sudo service docker restart 就OK了~~
或者 sudo service docker.io restart