黑马:https://www.bilibili.com/video/BV1HP4118797/?spm_id_from=333.337.search-card.all.click&vd_source=eca8bef44f4d5efc79f226fa688a5064
一、docker安装
1.1 安装
(一)安装命令 :
Ubuntu :
1 sudo apt install docker.io
(二)检查安装 :
查看示例 1 2 SDE@test-sg90-seatalk-10-136-185-3:~$ docker -v Docker version 26.1.3, build 26.1.3-0ubuntu1~24.04.1
1.2 启动和校验
(一)启动docker
查看示例 1 2 3 4 SDE@test-sg90-seatalk-10-136-185-3:~$ sudo systemctl start docker SDE@test-sg90-seatalk-10-136-185-3:~$ ps aux | grep docker root 247929 0.6 0.2 2135748 79460 ? Ssl 20:39 0:00 /usr/bin/dockerd -H fd:// --containerd=/run/containerd/containerd.sock SDE 248391 0.0 0.0 15272 2260 pts/0 S+ 20:40 0:00 grep --color=auto docker
(二)停止docker
查看示例 1 2 3 4 5 SDE@test-sg90-seatalk-10-136-185-3:~$ sudo systemctl stop docker Stopping 'docker.service', but its triggering units are still active: docker.socket SDE@test-sg90-seatalk-10-136-185-3:~$ ps aux | grep docker SDE 247495 0.0 0.0 15272 2316 pts/0 S+ 20:38 0:00 grep --color=auto docker
(三)重启docker
1 systemctl restart docker
(四)设置开机自启
关闭的话就是将enable
替换为disable
(五)查看docker服务状态
示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 SDE@test-sg90-seatalk-10-136-185-3:~$ sudo systemctl status docker sudo: unable to resolve host test-sg90-seatalk-10-136-185-3: Name or service not known ● docker.service - Docker Application Container Engine Loaded: loaded (/usr/lib/systemd/system/docker.service; enabled; preset: enabled) Active: active (running) since Wed 2025-05-28 20:39:42 +08; 6min ago TriggeredBy: ● docker.socket Docs: https://docs.docker.com Main PID: 247929 (dockerd) Tasks: 12 Memory: 26.9M () CPU: 365ms CGroup: /system.slice/docker.service └─247929 /usr/bin/dockerd -H fd:// --containerd=/run/containerd/containerd.sock 5月 28 20:39:41 test-sg90-seatalk-10-136-185-3 systemd[1]: Starting docker.service - Docker Application Container Engine... 5月 28 20:39:41 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:41.640004126+08:00" level=info msg="Starting up" 5月 28 20:39:41 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:41.670726588+08:00" level=info msg="[graphdriver] using prior storage driver: overlay2" 5月 28 20:39:41 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:41.672824181+08:00" level=info msg="Loading containers: start." 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:42.127423804+08:00" level=info msg="Default bridge (docker0) is assigned with an IP address 172.17.0.0/16. Daemon option --bip can be used to set a preferred IP address" 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:42.218058012+08:00" level=info msg="Loading containers: done." 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:42.228705548+08:00" level=info msg="Docker daemon" commit="26.1.3-0ubuntu1~24.04.1" containerd-snapshotter=false storage-driver=overlay2 version=26.1.3 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:42.228765294+08:00" level=info msg="Daemon has completed initialization" 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 dockerd[247929]: time="2025-05-28T20:39:42.254798664+08:00" level=info msg="API listen on /run/docker.sock" 5月 28 20:39:42 test-sg90-seatalk-10-136-185-3 systemd[1]: Started docker.service - Docker Application Container Engine.
Loaded: loaded (/usr/lib/systemd/system/docker.service; enabled; preset: enabled)
,第一个出现的enable就是表示开机自启
(六)查看正在运行的docker容器
查看示例 1 2 3 SDE@test-sg90-seatalk-10-136-185-3:~$ sudo docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES ddc9a2b86d1a mysql "docker-entrypoint.s…" 31 minutes ago Up 31 minutes 0.0.0.0:3306->3306/tcp, :::3306->3306/tcp, 33060/tcp mysql
1.3 配置镜像加速
由于使用的是公式的服务器,这里先跳过了
二、快速入门
2.1 安装mysql
1 2 3 4 5 6 docker run -d \ --name mysql \ -p 3306:3306 \ -e TZ=Asia/Shanghai \ -e MYSQL_ROOT_PASSWORD=123456 \ mysql
查看示例 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 SDE@test-sg90-seatalk-10-136-185-3:~$ sudo docker run -d --name mysql -p 3306:3306 -e TZ=Asia/Shanghai -e MYSQL_ROOT_PASSWORD=123456 mysql sudo: unable to resolve host test-sg90-seatalk-10-136-185-3: Name or service not known Unable to find image 'mysql:latest' locally latest: Pulling from library/mysql c2eb5d06bfea: Pull complete ba361f0ba5e7: Pull complete 0e83af98b000: Pull complete 770e931107be: Pull complete a2be1b721112: Pull complete 68c594672ed3: Pull complete cfd201189145: Pull complete e9f009c5b388: Pull complete 61a291920391: Pull complete c8604ede059a: Pull complete Digest: sha256:2247f6d47a59e5fa30a27ddc2e183a3e6b05bc045e3d12f8d429532647f61358 Status: Downloaded newer image for mysql:latest ddc9a2b86d1a453c3dd73dea70219d2e082569731f8f108e882f0c0e9c07dacf
连接测试:
大家可以发现,当我们执行命令后,Docker做的第一件事情,是去自动搜索并下载了MySQL,然后会自动运行MySQL,我们完全不用插手,是不是非常方便。
而且,这种安装方式你完全不用考虑运行的操作系统环境,它不仅仅在CentOS系统是这样,在Ubuntu系统、macOS系统、甚至是装了WSL的Windows下,都可以使用这条命令来安装MySQL。
要知道,不同操作系统下其安装包、运行环境是都不相同的 !如果是手动安装,必须手动解决安装包不同、环境不同的、配置不同的问题 !
而使用Docker,这些完全不用考虑。就是因为Docker会自动搜索并下载MySQL。注意:这里下载的不是安装包,而是镜像。镜像中不仅包含了MySQL本身,还包含了其运行所需要的环境、配置、系统级函数库。因此它在运行时就有自己独立的环境,就可以跨系统运行,也不需要手动再次配置环境了。这套独立运行的隔离环境我们称为 容器 。
说明:
镜像:英文是image
容器:英文是container
因此,Docker安装软件的过程,就是自动搜索下载镜像,然后创建并运行容器的过程。
Docker会根据命令中的镜像名称自动搜索并下载镜像,那么问题来了,它是去哪里搜索和下载镜像的呢?这些镜像又是谁制作的呢?
Docker官方提供了一个专门管理、存储镜像的网站,并对外开放了镜像上传、下载的权利。Docker官方提供了一些基础镜像,然后各大软件公司又在基础镜像基础上,制作了自家软件的镜像,全部都存放在这个网站。这个网站就成了Docker镜像交流的社区:https://hub.docker.com/
基本上我们常用的各种软件都能在这个网站上找到,我们甚至可以自己制作镜像上传上去。
像这种提供存储、管理Docker镜像的服务器,被称为DockerRegistry,可以翻译为镜像仓库。DockerHub网站是官方仓库,阿里云、华为云会提供一些第三方仓库,我们也可以自己搭建私有的镜像仓库。
官方仓库在国外,下载速度较慢,一般我们都会使用第三方仓库提供的镜像加速功能,提高下载速度。而企业内部的机密项目,往往会采用私有镜像仓库。
总之,镜像的来源有两种:
基于官方基础镜像自己制作
直接去DockerRegistry下载
总结一下 :
Docker本身包含一个后台服务,我们可以利用Docker命令告诉Docker服务,帮助我们快速部署指定的应用。Docker服务部署应用时,首先要去搜索并下载应用对应的镜像,然后根据镜像创建并允许容器,应用就部署完成了。
用一幅图标示如下:
2.2 安装命令解读
利用Docker快速的安装了MySQL,非常的方便,不过我们执行的命令到底是什么意思呢?
1 2 3 4 5 6 docker run -d \ --name mysql \ -p 3306:3306 \ -e TZ=Asia/Shanghai \ -e MYSQL_ROOT_PASSWORD=123456 \ mysql
解读:
docker run -d
:创建 并运行 一个容器,-d
则是让容器以后台进程运行
--name`` mysql
: 给容器起个名字叫mysql
,你可以叫别的
-p 3306:3306
: 设置端口映射。
-p <宿主机端口>:<容器端口>
容器是隔离环境 ,外界不可访问。但是可以将宿主机端口映射容器内到端口 ,当访问宿主机指定端口时,就是在访问容器内的端口了。
容器内端口往往是由容器内的进程决定,例如MySQL进程默认端口是3306,因此容器内端口一定是3306;而宿主机端口则可以任意指定,一般与容器内保持一致。
格式: -p 宿主机端口:容器内端口
,示例中就是将宿主机的3306映射到容器内的3306端口
-e TZ=Asia/Shanghai
: 配置容器内进程运行时的一些参数
格式:-e KEY=VALUE
,KEY和VALUE都由容器内进程决定
案例中,TZ=Asia/Shanghai
是设置时区;MYSQL_ROOT_PASSWORD=123456
是设置MySQL默认密码
mysql
: 设置镜像 名称,Docker会根据这个名字搜索并下载镜像
格式:REPOSITORY[:TAG]
,例如mysql:8.0
,其中REPOSITORY
可以理解为镜像名,TAG
是版本号
在未指定TAG
的情况下,默认是最新版本,也就是mysql:latest
镜像的名称不是随意的,而是要到DockerRegistry中寻找,镜像运行时的配置也不是随意的,要参考镜像的帮助文档,这些在DockerHub网站或者软件的官方网站中都能找到。
如果我们要安装其它软件,也可以到DockerRegistry中寻找对应的镜像名称和版本,阅读相关配置即可。
ps: 在 Docker 容器中,通常建议一个容器运行一个服务
三、docker基础
3.1 常见命令
首先我们来学习Docker中的常见命令,可以参考官方文档:
https://docs.docker.com/engine/reference/commandline/cli/
3.1.1 命令介绍
其中,比较常见的命令有:
用一副图来表示这些命令的关系:
补充:
默认情况下,每次重启虚拟机我们都需要手动启动Docker和Docker中的容器。通过命令可以实现开机自启:
1 2 3 4 5 # Docker开机自启 systemctl enable docker # Docker容器开机自启 docker update --restart=always [容器名/容器id]
3.1.2 演示
教学环节说明:我们以Nginx为例给大家演示上述命令。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 # 第1步,去DockerHub查看nginx镜像仓库及相关信息 # 第2步,拉取Nginx镜像 docker pull nginx # 第3步,查看镜像 docker images # 结果如下: REPOSITORY TAG IMAGE ID CREATED SIZE nginx latest 605c77e624dd 16 months ago 141MB mysql latest 3218b38490ce 17 months ago 516MB # 第4步,创建并允许Nginx容器 docker run -d --name nginx -p 80:80 nginx # 第5步,查看运行中容器 docker ps # 也可以加格式化方式访问,格式会更加清爽 docker ps --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}" # 第6步,访问网页,地址:http://虚拟机地址 # 第7步,停止容器 docker stop nginx # 第8步,查看所有容器 docker ps -a --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}" # 第9步,再次启动nginx容器 docker start nginx # 第10步,再次查看容器 docker ps --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}" # 第11步,查看容器详细信息 docker inspect nginx # 第12步,进入容器,查看容器内目录 docker exec -it nginx bash # 或者,可以进入MySQL docker exec -it mysql mysql -uroot -p # 第13步,删除容器 docker rm nginx # 发现无法删除,因为容器运行中,强制删除容器(-f) docker rm -f nginx
3.1.3 命令别名(linux技巧)
给常用Docker命令起别名,方便 我们访问:
1 2 3 4 5 6 7 8 9 10 11 SDE@test-sg90-seatalk-10-136-185-3:~$ vim ~/.bashrc ... alias dps='docker ps --format "table {{.ID}}\t{{.Image}}\t{{.Ports}}\t{{.Status}}\t{{.Names}}"' -- 插入 -- SDE@test-sg90-seatalk-10-136-185-3:~$ source ~/.bashrc SDE@test-sg90-seatalk-10-136-185-3:~$ dps CONTAINER ID IMAGE PORTS STATUS NAMES 7d5525d11fbe nginx 0.0.0.0:80->80/tcp, :::80->80/tcp Up 12 hours nginx ddc9a2b86d1a mysql 0.0.0.0:3306->3306/tcp, :::3306->3306/tcp, 33060/tcp Up 13 hours mysql
3.2 数据卷
容器是隔离环境,容器内程序的文件、配置、运行时产生的容器都在容器内部,我们要读写容器内的文件非常不方便。大家思考几个问题:
如果要升级MySQL版本,需要销毁旧容器,那么数据岂不是跟着被销毁了?
MySQL、Nginx容器运行后,如果我要修改其中的某些配置该怎么办?
我想要让Nginx代理我的静态资源怎么办?
因此,容器提供程序的运行环境,但是程序运行产生的数据、程序运行依赖的配置都应该与容器解耦 。
3.2.1 什么是数据卷
数据卷(volume)是一个虚拟目录,是 容器内目录与宿主机目录 之间映射的桥梁。
以Nginx为例,我们知道Nginx中有两个关键的目录:
html
:放置一些静态资源
conf
:放置配置文件
如果我们要让Nginx代理我们的静态资源,最好是放到html
目录;如果我们要修改Nginx的配置,最好是找到conf
下的nginx.conf
文件。
但遗憾的是,容器运行的Nginx所有的文件都在容器内部。所以我们必须利用数据卷将两个目录与宿主机目录关联,方便我们操作。如图:
在上图中:
我们创建了两个数据卷:conf
、html
Nginx容器内部的conf
目录和html
目录分别与两个数据卷关联。
而数据卷conf和html分别指向了宿主机的/var/lib/docker/volumes/conf/_data
目录和/var/lib/docker/volumes/html/_data
目录
这样以来,容器内的conf
和html
目录就 与宿主机的conf
和html
目录关联起来,我们称为挂载 。此时,我们操作宿主机的/var/lib/docker/volumes/html/_data
就是在操作容器内的/usr/share/nginx/html/_data
目录。只要我们将静态资源放入宿主机对应目录,就可以被Nginx代理了。
/var/lib/docker/volumes
是默认路径
小提示 :
/var/lib/docker/volumes
这个目录就是默认的存放所有容器数据卷的目录,其下再根据数据卷名称创建新目录,格式为/数据卷名/_data
。
为什么不让容器目录直接指向宿主机目录呢 ?(为什么需要通过容器数据卷来中转一下映射规则)
因为直接指向宿主机目录就与宿主机强耦合了,如果切换了环境,宿主机目录就可能发生改变了。由于容器一旦创建,目录挂载就无法修改,这样容器就无法正常工作了。
但是容器指向数据卷,一个逻辑名称,而数据卷再指向宿主机目录,就不存在强耦合。如果宿主机目录发生改变,只要改变数据卷与宿主机目录之间的映射关系即可。
不过,我们通过由于数据卷目录比较深,不好寻找,通常我们也允许让容器直接与宿主机目录挂载而不使用数据卷 ,具体参考3.2.3小节。
3.2.2 数据卷命令
数据卷的相关命令有:
注意:容器与数据卷的挂载要在创建容器时配置,对于创建好的容器,是不能设置数据卷的。而且创建容器的过程中,数据卷会自动创建 。
教学演示环节 :演示一下nginx的html目录挂载
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 # 1.首先创建容器并指定数据卷,注意通过 -v 参数来挂载指定数据卷 # https://hub.docker.com/_/nginx 官方文档中的地址: /usr/share/nginx/html # -v [数据卷名]:[挂载目录] 数据卷不存在的话会自动创建 docker run -d --name nginx -p 80:80 -v html:/usr/share/nginx/html nginx # 2.然后查看数据卷 docker volume ls # 结果 DRIVER VOLUME NAME local 29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f local html # 3.查看数据卷详情 docker volume inspect html # 结果 [ { "CreatedAt": "2024-05-17T19:57:08+08:00", "Driver": "local", "Labels": null, "Mountpoint": "/var/lib/docker/volumes/html/_data", "Name": "html", "Options": null, "Scope": "local" } ] # 4.查看/var/lib/docker/volumes/html/_data目录 ll /var/lib/docker/volumes/html/_data # 可以看到与nginx的html目录内容一样,结果如下: 总用量 8 -rw-r--r--. 1 root root 497 12月 28 2021 50x.html -rw-r--r--. 1 root root 615 12月 28 2021 index.html # 5.进入该目录,并随意修改index.html内容 cd /var/lib/docker/volumes/html/_data vi index.html # 6.打开页面,查看效果 # 7.进入容器内部,查看/usr/share/nginx/html目录内的文件是否变化 docker exec -it nginx bash
教学演示环节 :演示一下MySQL的匿名数据卷
我们关注两部分内容,第一是.Config.Volumes
部分:
1 2 3 4 5 6 7 8 9 { "Config" : { "Volumes" : { "/var/lib/mysql" : { } } } }
可以发现这个容器声明了一个本地目录,需要挂载数据卷,但是数据卷未定义 。这就是匿名卷(自动创建的)。
然后,我们再看结果中的.Mounts
部分:
1 2 3 4 5 6 7 8 9 10 11 { "Mounts" : [ { "Type" : "volume" , "Name" : "29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f" , "Source" : "/var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data" , "Destination" : "/var/lib/mysql" , "Driver" : "local" , } ] }
可以发现,其中有几个关键属性:
Name:数据卷名称。由于定义容器未设置容器名,这里的就是匿名卷自动生成的名字,一串hash值。
Source:宿主机目录
Destination : 容器内的目录
上述配置是将容器内的/var/lib/mysql
这个目录,与数据卷29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f
挂载。于是在宿主机中就有了/var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data
这个目录。这就是匿名数据卷对应的目录,其使用方式与普通数据卷没有差别。
接下来,可以查看该目录下的MySQL的data文件:
1 ls -l /var/lib/docker/volumes/29524ff09715d3688eae3f99803a2796558dbd00ca584a25a4bbc193ca82459f/_data
注意:每一个不同的镜像,将来创建容器后内部有哪些目录可以挂载,可以参考DockerHub对应的页面
3.2.3.挂载本地目录或文件
可以发现,数据卷的目录结构较深,如果我们去操作数据卷目录会不太方便。在很多情况下,我们会直接将容器目录与宿主机指定目录挂载。挂载语法与数据卷类似:
1 2 3 4 # 挂载本地目录 -v 本地目录:容器内目录 # 挂载本地文件 -v 本地文件:容器内文件
注意 :本地目录或文件必须以 /
或 ./
开头,如果直接以名字开头,会被识别为数据卷名而非本地目录名。
例如:
1 2 -v mysql:/var/lib/mysql # 会被识别为一个数据卷叫mysql,运行时会自动创建这个数据卷 -v ./mysql:/var/lib/mysql # 会被识别为当前目录下的mysql目录,运行时如果不存在会创建目录
教学演示 ,删除并重新创建mysql容器,并完成本地目录挂载:(这些容器目录是从官网上查的)
挂载/root/mysql/data
到容器内的/var/lib/mysql
目录
挂载/root/mysql/init
到容器内的/docker-entrypoint-initdb.d
目录(初始化的SQL脚本目录)
挂载/root/mysql/conf
到容器内的/etc/mysql/conf.d
目录(这个是MySQL配置文件目录)
使用已经准备好了文件:
https://github.com/ggw2021/static/tree/main/learning/docker/mysql
其中,hm.cnf主要是配置了MySQL的默认编码,改为utf8mb4;而hmall.sql则是后面我们要用到的黑马商城项目(没做过,直接用就好了)的初始化SQL脚本。
我们直接将整个mysql目录上传至虚拟机的 自己指定的目录下(这里使用pwd: /home/toc/SDE
)
接下来,我们演示本地目录挂载:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 # 1.删除原来的MySQL容器 docker rm -f mysql # 2.进入/home/toc/SDE目录 cd /home/toc/SDE # 3.创建并运行新mysql容器,挂载本地目录 docker run -d \ --name mysql \ -p 3306:3306 \ -e TZ=Asia/Shanghai \ -e MYSQL_ROOT_PASSWORD=123 \ -v ./mysql/data:/var/lib/mysql \ -v ./mysql/conf:/etc/mysql/conf.d \ -v ./mysql/init:/docker-entrypoint-initdb.d \ mysql # 4.查看/home/toc/SDE目录,可以发现~/mysql/data目录已经自动创建好了 ls -l mysql # 结果: 总用量 4 drwxr-xr-x. 2 SDE toc 20 5月 19 15:11 conf drwxr-xr-x. 7 polkitd toc 4096 5月 19 15:11 data drwxr-xr-x. 2 SDE toc 23 5月 19 15:11 init # 查看data目录,会发现里面有大量数据库数据,说明数据库完成了初始化 ls -l data # 5.查看MySQL容器内数据 # 5.1.进入MySQL docker exec -it mysql mysql -uroot -p123 # 5.2.查看编码表 show variables like "%char%"; # 5.3.结果,发现编码是utf8mb4没有问题 +--------------------------+--------------------------------+ | Variable_name | Value | +--------------------------+--------------------------------+ | character_set_client | utf8mb4 | | character_set_connection | utf8mb4 | | character_set_database | utf8mb4 | | character_set_filesystem | binary | | character_set_results | utf8mb4 | | character_set_server | utf8mb4 | | character_set_system | utf8mb3 | | character_sets_dir | /usr/share/mysql-8.0/charsets/ | +--------------------------+--------------------------------+ # 6.查看数据 # 6.1.查看数据库 show databases; # 结果,hmall是黑马商城数据库 +--------------------+ | Database | +--------------------+ | hmall | | information_schema | | mysql | | performance_schema | | sys | +--------------------+ 5 rows in set (0.00 sec) # 6.2.切换到hmall数据库 use hmall; # 6.3.查看表 show tables; # 结果: +-----------------+ | Tables_in_hmall | +-----------------+ | address | | cart | | item | | order | | order_detail | | order_logistics | | pay_order | | user | +-----------------+ # 6.4.查看address表数据 +----+---------+----------+--------+----------+-------------+---------------+-----------+------------+-------+ | id | user_id | province | city | town | mobile | street | contact | is_default | notes | +----+---------+----------+--------+----------+-------------+---------------+-----------+------------+-------+ | 59 | 1 | 北京 | 北京 | 朝阳区 | 13900112222 | 金燕龙办公楼 | 李佳诚 | 0 | NULL | | 60 | 1 | 北京 | 北京 | 朝阳区 | 13700221122 | 修正大厦 | 李佳红 | 0 | NULL | | 61 | 1 | 上海 | 上海 | 浦东新区 | 13301212233 | 航头镇航头路 | 李佳星 | 1 | NULL | | 63 | 1 | 广东 | 佛山 | 永春 | 13301212233 | 永春武馆 | 李晓龙 | 0 | NULL | +----+---------+----------+--------+----------+-------------+---------------+-----------+------------+-------+ 4 rows in set (0.00 sec)