一、何为docker

docker最早基于LXC实现(LinuX Container)从0.7版本以后开始去除LXC转而使用自行开发的libcontainer,从1.11开始,演进为runC和containerd;docker是go语言开发,基于Linux内核的cgroup,namespace以及AUFS类似的Union FS(联合文件系统)等技术,对进程进行封装隔离,属于操作系统层面的虚拟化技术,由于隔离的进程独立于宿主机和其他的隔离进程,因此也称其为容器。 docker在容器的基础上进行了进一步的封装从文件系统,网络互联到进程隔离等,极大简化了容器的创建和维护管理。使得docker技术比虚拟机技术更为轻便,快捷。

虚拟机与docker比较: 图片引用自网络

由上面的比较图可以看同样启动一个服务;虚拟机会多一层操作系统,而docker和宿主机共用内核系统;因此docker占用资源更少,资源利用率更高; 虚拟机与docker对比

docker优点总结: 高效的资源利用率 快速的启动 一致的运行环境 持续将会和部署 更轻松的迁移 便捷的维护和扩展

二、dcoker安装

系统要求:CentOS7.x_x64且安装Docker CE版本(社区版),不要使用epel仓库自带的docker docker从1.17后主要分docker-ce和docker-ee(企业版,你懂滴,就好比mysql企业版与社区版,如果还不明白当我没说)

以下安装基于CentOS 7.4_x64 最小化安装

[root@docker ~]# uname -r
3.10.0-693.21.1.el7.x86_64
[root@docker ~]# cat /etc/redhat-release 
CentOS Linux release 7.4.1708 (Core) 

#御载自带组件
yum remove docker docker-client docker-client-latest  docker-common docker-latest docker-latest-logrotate docker-logrotate docker-selinux docker-engine-selinux  docker-engine -y
#安装依赖组件
[root@docker ~]# yum install -y yum-utils device-mapper-persistent-data  lvm2
#添加国内源
$ yum-config-manager --add-repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
或
$ yum-config-manager --add-repo https://mirrors.ustc.edu.cn/docker-ce/linux/centos/docker-ce.repo

#添加官方源
[root@docker ~]# yum-config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo

#最新版的docker CE 使用以下命令
[root@docker ~]# yum-config-manager --enable docker-ce-edge

#测试版docker ce 使用以下命令 (可选)
 [root@docker ~]# yum-config-manager --enable docker-ce-test

#安装docker ce
 [root@docker ~]# yum makecache fast
 [root@docker ~]# yum install install docker-ce
 #注意官方docker-ce可能有点慢~
 
#或使用脚本自动安装
 [root@docker ~] # curl -fsSL get.docker.com -o get-docker.sh
 [root@docker ~]# sh get-docker.sh --mirror Aliyun

#安装完查看版本信息
[root@docker ~]# docker --version
Docker version 18.04.0-ce, build 3d479c0

#启动docker
[root@docker ~]# systemctl start docker

以上表示安装成功docker-ce版本,以下的示例测试均在此基础上进行;如果是ubuntu安装docker请参考这里ubuntu安装docker-ce

三、docker组件

从物理上讲client <--> docker daemon <--> Registry Server 逻辑上: Images: 镜像只读,分层镜像机制,最后一层可读写,可共享,或理解为OS Containers:容器: 是从镜像启动创建的,类似程序与进程的关系,而Images镜像就是程序文件,运行起来成为进程;镜像是静态定义,容器是镜像运行时的实体;容器可以被创建,启动,停止,删除,暂停;容器实质是进程;

Registry:Image Repositories (存放共享的镜像库,类似yum库)docker官方提供仓库地址hub.docker.com

容器的状态与指令: 常用状态如: created:docker create(不常用),docker run 指令从镜像创建并运行 running:docker run 指令从镜像创建并运行 paused:docker paused 暂停容器 stopped:docker stop 停止运行某容器,或通过docker kill容器 状态间转化如图:

以上指令均可通过docker --help 或docker COMMAND --help获取子命令帮助

查看docker相关信息:

docker --version
Docker version 18.03.0-ce, build 0520e24

获取docker daemon运行的信息 docker info

镜像相关操作 列出本机镜像: docker images 拉取下载镜像: docker pull 库/镜像名:版本 默认是docker.io/镜像名:lastest 删除镜像: 删除关确保没有容器使用 docker rmi 镜像名|镜像ID 查找镜像: docker search 镜像名

[root@docker ~]# docker search busybox
NAME                        DESCRIPTION                                     STARS               OFFICIAL            AUTOMATED
busybox                     Busybox base image.                             1239                [OK]                
progrium/busybox                                                            66                                      [OK]

说明:STARS可理解为获取的赞,OFFICIAL OK表示官方出的

示例:拉取busybox

[root@docker ~]# docker pull busybox
Using default tag: latest

注意:没有指定仓库,默认从docker.io上获取最新latest版本

容器相关常用操作 docker run 选项 镜像:创建并运行一个容器 选项如下: --rm 运行容器停止退出后删除容器 --it 交互式运行容器 -d 后台daemon运行(和--rm不可同时出现) --name 指定容器名 -p [port:port] 把容器中的端口映射出来

docker ps :查看运行中的容器 docker ps -a:查看所有的容器(包括停止状态) docker start :启动处于停止状态的容器 docker inspect 容器:查运行中容器信息 docker network ps :查看docker 网络接口 docker attach:附加至某运行状态的容器的终端设备;(运行在后台中的容) docker exec:让运行中的容器运行一个额外的程序(容器中有程序前台占终端运行情况下,后面有实例) docker logs:输出容器内部程序运行时输出到终端的信息 docker stats:动态方式显示容器的资源占用状态 docker top:显示容器运行的进程 docker kill :停止容器(容器牌前台运行时) docker stop:停止容器 docker tag 镜像 新镜像:标记镜像 docker login -u 用户名: 登录(默认docker.io需要注册)用户名 docker pull 镜像:默认从docker.io仓库获取最新版本镜像,如是私有仓库需要登录 docker push 镜像:登录后可push(默认docker.io)仓库 docker rename:重命名容器 docker restart: 重启容器 docker pause:暂停容器 docker unpause:从暂停状态恢复 docker container update: 对现有容器启动参数更新修性 docker rm :删除容器,需要注意的是容器只能是关闭状态才能删除 或运行进docker run --rm 表示运行停止后就删除容器 更多docker命令行参考

示例:从busybox镜像启动容器b1 前台运行

[root@docker ~]# docker run --rm -it --name b1 busybox
#
按crtl + p +q  从终端拆除(转入后台)
#查看容器
[root@docker ~]# docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED              STATUS              PORTS               NAMES
356979c81de8        busybox             "sh"                About a minute ago   Up About a minute                       b1

说明:可以看到运行状态是up  容器名指定为b1 可以通过docker stop b1 停止 (由于 --rm选项)即容器删除
[root@docker ~]# docker stop b1
b1
[root@docker ~]# docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
[root@docker ~]# 

示例:后台启动容器 b1

[root@docker ~]# docker run -d -it --name b1 busybox
b8548d53a69999b0c351dc779ebf18cdcfd40dc4d749bc7a6cec300d3ad958e5
[root@docker ~]# docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
b8548d53a699        busybox             "sh"                17 seconds ago      Up 16 seconds                           b1
[root@docker ~]# docker stop b1
b1
[root@docker ~]# docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS                        PORTS               NAMES
b8548d53a699        busybox             "sh"                49 seconds ago      Exited (137) 14 seconds ago                       b1

更新busybox开机自启
docker container update --restart=always busybox

说明:通过后台启动模式,在停止运行后 容器处于退出停止状态,并不会删除,下次可以通过 docker start b1再次启动,--rm和-d不能同时存在。

四、镜像制作

镜像制作有两种方式,基于容器制作和基于(已有)镜像制作; 基于(运行中)容器制作 在容器中完成操作后制作 docker commit [OPTIONS] CONTAINER [REPOSITORY[:TAG] 示例:基于busybox 镜像启动容器b1并在容器里启动httpd服务 并以此运行的容器制作镜像

运行容器(进入容器操作)
[root@docker ~]# docker run -it --name b1 busybox
/ # mkdir -pv /data/html
created directory: '/data/'
created directory: '/data/html'
/ # echo " Busybox httpd server" >/data/html/index.html
/ # cat /data/html/index.html 
 Busybox httpd server
/ # httpd -h /data/html
/ # ifconfig 
eth0      Link encap:Ethernet  HWaddr 02:42:AC:11:00:02  
          inet addr:172.17.0.2  Bcast:172.17.255.255  Mask:255.255.0.0
						....省略....
/ # netstat -ntpul
Active Internet connections (only servers)
Proto Recv-Q Send-Q Local Address           Foreign Address         State       PID/Program name    
tcp        0      0 :::80                   :::*                    LISTEN      8/httpd

按ctrl +p +q 进入后台(拆除终端)
[root@docker ~]# docker ps -a
CONTAINER ID        IMAGE               COMMAND             CREATED             STATUS              PORTS               NAMES
477cf5b3abca        busybox             "sh"                4 minutes ago       Up 4 minutes                            b1

从以上可以看出在b1容器中启动了http 并侦听在80端口上;拆除终端进入了后台运行;

基于运行中的b1容器制作镜像并打上新的标签名"docker.io/dyc2005/bbox:v0.1.1-httpd"

[root@docker ~]# docker commit -p -a "san <san@san.com>"  b1 docker.io/dyc2005/bbox:v0.1.1-httpd
sha256:c4e8c59c4b577a568002789e7c266828a8e86dd9fdd825b05f91933a5e9710f0
[root@docker ~]# docker images
REPOSITORY          TAG                 IMAGE ID            CREATED             SIZE
dyc2005/bbox        v0.1.1-httpd        c4e8c59c4b57        6 seconds ago       1.15MB
busybox             latest              8ac48589692a        9 days ago          1.15MB

此时新的镜像ID和busybox(产生b1容器)镜像ID已经 不同了,此时也可通过 docker push 推到镜像库中 验证: 基于新的镜像后台启动容器b2 补充:与上面效果类似 [root@docker ~]# docker commit -a "san" -c 'CMD ["/bindocke/httpd","-f","-h","/data/html"]' b1 docker.io/dyc2005/bbox:v0.1.2-httpd #运行制作好的镜像: [root@docker ~]# docker run --name h2 -it dyc2005/bbox:v0.1.2-httpd [root@docker ~]# docker exec h2 ps PID USER TIME COMMAND 1 root 0:00 /bin/httpd -f -h /data/html 8 root 0:00 ps [root@docker ~]# docker ps -a CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES c7147cb559d9 dyc2005/bbox:v0.1.3-httpd "/bin/httpd -f -h /d…" 22 seconds ago Up 21 seconds h2

[root@docker ~]# docker run -it -d  --name b2 docker.io/dyc2005/bbox:v0.1.1-httpd httpd -f -h /data/html
b93754954d61b1a477ef361fcb348962b9b10004221263d9f4c69db193730a61
#查看b2的信息查看ip
[root@docker ~]# docker inspect b2 |grep IPAddress
            "SecondaryIPAddresses": null,
            "IPAddress": "172.17.0.3",
                    "IPAddress": "172.17.0.3",
#访问b2
[root@docker ~]# curl http://172.17.0.3
 Busybox httpd server

至此基于运行中的容器制作镜像完成! 此时产生一个问题,运行中的镜像进入是docker attach b2 但你发现终端卡在那里,由于httpd(容器中)前台运行,无法修改网页 解决方法:

[root@docker ~]# docker exec -it b2 /bin/sh
/ # netstat -ntpul
Active Internet connections (only servers)
Proto Recv-Q Send-Q Local Address           Foreign Address         State       PID/Program name    
tcp        0      0 :::80                   :::*                    LISTEN      1/httpd
/ # 

通过 docker exec 就可以进行运行在容器前端的中修改容器中的内容!这也是attach与exec命令的区别所在! 另外本次运行镜像端口没有做映射,后面会提到!一般不推荐如此制作镜像;制作镜像推荐使用Dockerfile文件

基于镜像制作 编辑 Dockerfile,而后根据此文件制作; 详细说明参照这里 以下给出实例: 宿主机上创建bbox [root@docker ~]# mkdir -pv bbox/data [root@docker ~]# cat /data/html/index.html Busybox httpd server from Dockerfile. [root@docker ~]#cat Dockerfiel

FROM busybox:latest
MAINTAINER "san" "<san@san.com>"
#COPY index.html /data/html/
COPY data /data/
VOLUME /data/html
EXPOSE 80/tcp
ENV epel_repo_file http://172.16.0.188/epel.repo
ADD ${epel_repo_file} /etc/yum.repos.d/
RUN /bin/adduser -D myuser && \
    /bin/mkdir /tmp/dir -pv

注意请自行模拟一个可访问的http能访问到的epel.repo 在bbox目录中运行

[root@docker bbox]# docker build -t docker.io/dyc2005/busybox/v.1.1-httpd ./
Sending build context to Docker daemon  4.096kB
Step 1/8 : FROM busybox:latest
 ---> 8ac48589692a
Step 2/8 : MAINTAINER "san" "<san@san.com>"
 ---> Using cache
 ---> 3105f31095b2
Step 3/8 : COPY data /data/
 ---> Using cache
 ---> 355f7a29fd45
Step 4/8 : VOLUME /data/html
 ---> Using cache
 ---> 4111c015d683
Step 5/8 : EXPOSE 80/tcp
 ---> Using cache
 ---> 6f2ef2457118
Step 6/8 : ENV epel_repo_file http://172.16.0.188/epel.repo
 ---> Using cache
 ---> d1ac10e625c4
Step 7/8 : ADD ${epel_repo_file} /etc/yum.repos.d/
Downloading [==================================================>]      10B/10B
 ---> 943a7f280b6e
Step 8/8 : RUN /bin/adduser -D myuser &&     /bin/mkdir /tmp/dir -pv
 ---> Running in 15bb73b0690d
created directory: '/tmp/dir'
Removing intermediate container 15bb73b0690d
 ---> e2ff30f1beb2
Successfully built e2ff30f1beb2
Successfully tagged dyc2005/busybox/v.1.1-httpd:latest

##查看新生成的镜像并运行
[root@docker bbox]# docker images
REPOSITORY                    TAG                 IMAGE ID            CREATED             SIZE
dyc2005/busybox/v.1.1-httpd   latest              e2ff30f1beb2        52 seconds ago      1.15MB
[root@docker bbox]# docker run -it --rm --name d1 dyc2005/busybox/v.1.1-httpd
/ # id myuser
uid=1000(myuser) gid=1000(myuser) groups=1000(myuser)
/ # ls /tmp/dir/
/ # ls /tmp/
dir
/ # 
可以看出新生成的镜像已经完成了指定的操作

到此镜像制作的两咱方法已经小试牛刀的演示完,推荐用dockerfile文件生成镜像,Dockerfile中的每一行的命令都是一层 因此建议使用command && command && ...这种方式;

五、镜像的导入与导出以及使用Volume(外部的目录)

将镜像文件导出为tar文件 docker save #镜像保存成文件(在没有仓库的情况下) Save one or more images to a tar archive (streamed to STDOUT by default) docker save [OPTIONS] IMAGE [IMAGE...] 示例:docker save -o /root/nginx-v0.1.0.tar 4481eeccc973

**从tar文件导入镜像 **

docker load ##从文件导入成镜像
Load an image from a tar archive or STDIN            
docker load [OPTIONS]            
    --input, -i	 	Read from tar archive file, instead of STDIN
    --quiet, -q	false	Suppress the load output	
    示例
# docker load -i nginx-v0.1.0.tar

重新打标签 [root@docker ~]# docker tag 4481eeccc973 docker.io/dyc2005/nginx.v0.1.0

Bind mount volume :容器内和宿主机上都是用户指定
[root@docker ~]# run -it --name bbox1 -v /data/html  busybox:v1.0
[root@docker ~]# cat /var/lib/docker/volumes/163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05/_data# cat index.html 
           c1        
[root@docker ~]#docker inspect -f {{.Config.Volume}}{{.Mounts}} bbox1
Docker-managed volume :容器内用户指定,宿主机上的目录默认在/var/lib/docker/volumes 由docker指定
[root@docker ~]#docker run -it -v HOSTDIR:VOLUME --name bbox2  busybox:v1.0
[root@docker ~]# mkdir -pv /tmp/data(宿主机上)
[root@docker ~]# echo " Test Page in App1" >/tmp/data/index.html
[root@docker ~]# docker run --name h2 -d -it  -v /tmp/data:/data/html dyc2005/bbox:v0.1.3-httpd
[root@docker ~]# curl http://172.17.0.2
        Test Page in App1
[root@docker ~]# docker inspect -f {{.Mounts}} bbox2

列出volume卷:

[root@docker ~]#docker inspect -f {{.Mounts}} bbox2 docker volume list DRIVER VOLUME NAME local 163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05

查看详细信息

[root@docker ~]#docker inspect -f {{.Mounts}} bbox2
docker volume inspect
163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05
    [
        {
            "CreatedAt": "2018-04-09T18:10:46+08:00",
            "Driver": "local",
            "Labels": null,
            "Mountpoint": "/var/lib/docker/volumes/163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05/_data",
            "Name": "163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05",
            "Options": {},
            "Scope": "local"
        }
    ]
    或
[root@docker ~]#  inspect -f {{.Mounts}} bbox2
    [{volume 163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05 /var/lib/docker/volumes/163a16717cb1d859869cdd7a87a3ad91b04e754ff26130d9145fe0c4ccf94e05/_data /data/html local  true }]

注意问题:挂载volume到容器,如果容器中有内容则复制到宿主机目录,一般删除容器不会删除volume,如果要一并删除需要加--volumw|-v选项 单独删除 docker volume rm 命令

多个容器使用同一个宿主目录

[root@docker ~]# docker run --name h1 -d -it  -v /tmp/data:/data/html dyc2005/bbox:v0.1.3-httpd
[root@docker ~]# docker run --name h2 -d -it  -v /tmp/data:/data/html dyc2005/bbox:v0.1.3-httpd
或复制其他主机使用的volume
[root@docker ~]# docker run --name h1 -d -it  -v /tmp/data:/data/html dyc2005/bbox:v0.1.3-httpd
        
 [root@docker ~]# docker run --name h3 -it --rm --volumes-from h1 dyc2005/bbox:v0.1.0
    #cd /data/html/
    /data/html # ls
    index.html
        

补充: 一、容器里执行systemctl 命令出错

Failed to get D-Bus connection: Operation not permitted

解决方案一: docker run -d -e "container=docker" --privileged=true -v /sys/fs/cgroup:/sys/fs/cgroup --name centos7 centos /usr/sbin/init ###对安全问题比较关心的同学,建议将--privileged=true替换为--cap-add=SYS_ADMIN 方案二: docker run -ti --privileged centos init #或者 docker run -ti --privileged centos /usr/lib/systemd/systemd #或者 docker run -ti --cap-add=SYS_ADMIN cents init

如果是已经运行中的容器安装了如mysqld服务执行systemctl报以上错误;则重新从容器打成镜像,再从新镜像启动一个mysqld容器 如: docker commit -p -a "test test@san.com" exciting_dirac centos/mysql:mysql5.7 说明: exciting_dirac容器是运行中的容器,且安装了mysql服务,通过以上命令重新生成镜像centos/mysql:mysql5.7

[root@study01 ~]# docker images
REPOSITORY                TAG        IMAGE ID       CREATED          SIZE
centos/mysql              mysql5.7   fe3c32f11a3e   13 minutes ago   1.77GB

再次启动一个含有mysql服务的容器:

[root@study01 ~]# docker run -ti -d  --name mysql --privileged centos/mysql:mysql5.7 /usr/lib/systemd/systemd

[root@study01 ~]# docker ps -a
CONTAINER ID   IMAGE                     COMMAND                  CREATED          STATUS         PORTS     NAMES
cfe3e007ead9   centos/mysql:mysql5.7     "/usr/lib/systemd/sy…"   15 minutes ago   Up 7 minutes  mysql

更新mysql服务开机自启
docker container update --restart=always mysql
二、查看容器pid
PID=$(docker inspect --format '{{.State.Pid}}' mysql)
echo $PID
1401

以上是docker学习中整理的笔记难免有错漏之处,欢迎指正!