MySQL联合查询

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: MySQL联合查询

一、内连接查询

image.png

先创建表student、course、exam


CREATE TABLE student(
  uid INT UNSIGNED PRIMARY KEY NOT NULL AUTO_INCREMENT,
  name VARCHAR(20) NOT NULL,
  age TINYINT UNSIGNED NOT NULL,
  sex enum("man", "woman") NOT NULL
);
CREATE TABLE course(
  cid INT UNSIGNED PRIMARY KEY NOT NULL AUTO_INCREMENT,
  name VARCHAR(20) NOT NULL,
  credit TINYINT UNSIGNED NOT NULL
);
CREATE TABLE exam(
  uid INT UNSIGNED NOT NULL,
  cid INT UNSIGNED NOT NULL,
  time DATE NOT NULL,
  score FLOAT NOT NULL,
  PRIMARY KEY(uid, cid)
);
insert into student(name, age, sex) values('zhangsan',18,'man'),('gaoyang',20,'woman'),('chenwei',22,'man'),('linfeng',21,'woman'),('liuxiang',19,'man');
insert into course(cname, credit) values('C++基础课程', 5),('C++高级课程', 10),('C++项目开发', 8),('C++算法课程', 12);
insert into exam(uid,cid,time,score) values(1,1,20210409,99),(1,2,20210410,80),(2,2,20210410,90),(2,3,20210412,99),(3,1,20210409,56),(3,2,20210410,93),(3,3,20210412,89),(3,4,20210411,100),(4,4,20210411,99),(5,2,20210410,59),(5,3,20210412,94),(5,4,20210411,95);

image.png

我们现在想查询1号同学的个人信息以及他的2号课程的成绩

  1. 先查询在学生表中查询zahngsan的详细信息:


select uid, name, age, sex from student where uid=1;
  1. 然后再考试表中查询zhangsan同学的考试信息:


select time, score from exam where uid=1 and cid=2;
  1. 内连接合并两次查询的结果

由于是查询学生表和考试表公共的部分,索引使用inner join


select stu.uid, stu.name, stu.age, stu.sex, ex.time, ex.score from student as stu 
inner join exam as ex 
on stu.uid=ex.uid 
where ex.uid=1 and ex.cid=2;

image.png

  1. 上述查询还看不见课程名,我们再次使用内连接,连接3张表,查询带有课程名的信息


select stu.uid, stu.name, stu.age, stu.sex, ex.time, ex.score, co.cname from exam as ex -- 第一个放用于连接的中间表,即和另外两张表有公共部分的表
inner join student as stu on ex.uid=stu.uid -- 连接考试表和学生表
inner join course as co on ex.cid=co.cid    -- 连接考试表和课程表
where ex.uid=1 and ex.cid=2;                -- 过滤条件

image.png

  1. 查询每门课高于90分的有多少个人


select course.cid, course.cname, count(*) cnt
from course
join exam on exam.cid=course.cid  -- 连接
where exam.score > 90.0           -- 条件过滤
group by exam.cid                 -- 分组
order by cnt desc;                -- 排序

image.png

  1. 查询cid=2这门课程考最高分的学生信息和课程信息


select co.cname, co.credit, stu.name, ex.score
from exam ex
inner join course co on ex.cid=co.cid
inner join student stu on ex.uid=stu.uid
where ex.cid=2
order by ex.score desc
limit 1;

image.png

  1. 查询cid=2这门课程的平均成绩


select avg(score) from exam where cid=2;
  1. 每门课程的详细信息以及平均成绩


select co.cname, co.credit, avg(score) from course co
inner join exam ex
on co.cid=ex.cid
group by ex.cid;

image.png

在MySQL库表操作以及简单查询语句中有提到,可以使用select属性的数量也会影响查询的速度,也可以使用条件过滤where <带有索引的属性>加快查询,现在我们介绍使用内连接优化查询

image.png

能不能在查询多个属性的情况下,还能花费较少的时间呢?

内连接优化查询

优化原理:由于生成小表(临时表)的时候使用了带有索引的属性id,故生成小表很快,接着用小表的数据在大表t_user里面匹配id,也使用了索引,故能加快查询


select a.id, a.email, a.password from
t_user inner join (select id from t_user limit 1500000, 10) b
on a.id=b.id

image.png

二、详细解释内连接的查询过程

数据库引擎如何按照 on a.uid=b.uid 进行表合并的?

先根据数据量确定大表和小表,小表永远是整表扫描,拿着小表的uid去大表搜索uid。很明显,由于小表永远是整表扫描,无需使用索引,我们一般给大表建索引加快搜索。

image.png

对于inner join而言,假设一开始A表是大表,B表是小表,数据库引擎拿着B表的所有数据去A表做匹配的时候,发现SQL语句还有where,这时候就需要进行数据过滤,过滤出满足条件的数据。此时可能由于A表满足条件的数据比B表满足条件的数据还少,这是A表满足条件的数据形成的表成了小表,B表满足条件的数据形成的表成了大表。总结下来就是先用where进行数据过滤,在用小表的数据去大表匹配满足on条件的数据

image.png

对于inner join,where的子条件放在on后面,效果和效率是一样的。因为MySQL引擎会把on后面的条件优化为where,where是可以使用索引的,效率高。

image.png

三、左、右连接

image.png

外连接不区分大小表,只有inner join区分大小表


// left join把左表的所有数据显示出来,若右表不存在,则显示为NULL
select * from student left join exam on student.uid=exam.uid;
// right join把右表的所有数据显示出来,若左表不存在,则显示为NULL
select * from student right join exam on student.uid=exam.uid;

image.png

image.png

内连接区分大小表,外连接不区分大小表。对于左右连接而言,都有一个表需要整表搜索

image.png

应用场景1:查看没有参加考试的同学


-- select distinct uid from exam 会产生一张中间表供外面的SQL查询
-- not in对索引的命中并不高,一般情况下都是整表扫描
select * from student where uid not in (select distinct uid from exam);
-- 使用外连接,只显示课程号为null的学生信息
select a.* from student a left join exam b on a.uid=b.uid where b.cid is null;

image.png

在一些没有出现的场景的时候经常使用外连接,比如没有出现在任何订单的商品等

应用场景2:查看没有参加3号课程考试的同学


-- 查看参加了3号课程的人,用where过滤后,再决定大小表关系,大表整表扫描
select a.* from student a inner join exam b on a.uid=b.uid where b.cid=3;
-- 看起来是left join,其实变成了inner join,查询过程和inner join一样
select a.* from student a left join exam b on a.uid=b.uid where b.cid=3;

image.png

on后面写连接条件,where后面写过滤条件


-- 查看参加了考试同学的信息,未参加3号课程考试的则用NULL填充
select a.*, b.* from student a left join exam b on a.uid=b.uid and b.cid=3;
-- 查看没有参加3号课程考试的同学
select a.*, b.* from student a left join exam b on a.uid=b.uid and b.cid=3 where b.cid is null;

image.png

image.png

左连接:数据量和左表相同,无法连接的部分用null填充


select * from Customers left join Orders on Customers.Id = Orders.CustomerId;
id name id customerId
1 “Joe” 2 1
2 “Henry” null null
3 “Sam” 1 3
4 “Max” null null

内连接


select * from Customers join Orders on Customers.Id = Orders.CustomerId;
id name id customerId
1 “Joe” 2 1
3 “Sam” 1 3

右连接:数据量和右表相同,无法连接的部分用null填充


select * from Customers right join Orders on Customers.Id = Orders.CustomerId;
id name id customerId
3 “Sam” 1 3
1 “Joe” 2 1


相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
关系型数据库 MySQL
Mysql union 联合查询
Mysql union 联合查询
85 0
|
存储 关系型数据库 MySQL
MySQL数据库的跨库查询和联合查询技巧
MySQL数据库的跨库查询和联合查询技巧
|
2月前
|
NoSQL 安全 关系型数据库
2024Mysql And Redis基础与进阶操作系列(6)作者——LJS[含MySQL 多表之一对一/多;多对多;多表联合查询等详解步骤及常见报错问题所对应的解决方法]
MySQL 多表之一对一/多;多对多;多表联合之交叉连接;内连接;左、右、外、满、连接;子查询及关键字;自连接查询等详解步骤及常见报错问题所对应的解决方法
|
7月前
|
Oracle 关系型数据库 MySQL
MYSQL-多表联合查询汇总
MYSQL-多表联合查询汇总
|
7月前
|
SQL 关系型数据库 MySQL
MySQL数据库——多表查询(3)-自连接、联合查询、子查询
MySQL数据库——多表查询(3)-自连接、联合查询、子查询
417 1
|
7月前
|
关系型数据库 MySQL 数据库
Mysql数据库联合查询
Mysql数据库联合查询
108 0
|
7月前
|
SQL 关系型数据库 MySQL
MySQL多表联合查询+分组+排序
MySQL多表联合查询+分组+排序
42 0
|
8月前
|
存储 SQL 关系型数据库
MySQL表的增删改查---多表查询和联合查询
MySQL表的增删改查---多表查询和联合查询
|
8月前
|
SQL 存储
MySQL-多表联合查询
MySQL-多表联合查询
|
8月前
|
关系型数据库 MySQL 数据库
MySql数据库联合查询(MySql数据库学习——六)
MySql数据库联合查询(MySql数据库学习——六)
92 0