联合查询*


1、什么是联合查询:就是两张表或者两张以上的表,进行连接查询。
2、为什么要联合查询:就是因为,我们所需要的数据,不仅仅是来自于一张表的,他是来自于多张表的。所以,我们要进行联合查询!!!
3、前置知识:笛卡尔积!!! 排列组合而已!

笛卡尔积

在这里插入图片描述
所有的联合查询,都是从这个笛卡尔积当中去取出数 据!!!当然在取数据的时候,一定是要满足某种规则的。 这种规则就是我们接下来学习的各种连表查询的方式!!!
准备数据:

drop database if exists test0311; 
create database test0311; 
use test0311; 
drop table if exists classes; 
create table classes( 
       id int primary key auto_increment, 
       name varchar(50), 
       `desc` varchar(50) 
     );
insert into classes(name, `desc`) values 
('计算机系2019级1班', '学习了计算机原理、C和Java语言、数据结构和算法'), 
('中文系2019级3班','学习了中国传统文学'), 
('自动化2019级5班','学习了机械自动化'); 

drop table if exists student; 
create table student( 
     id int primary key auto_increment, 
     sn int,
     name varchar(30), 
     qq_mail varchar(30), 
     classes_id int 
   );
insert into student(sn, name, qq_mail, classes_id) values 
('09982','黑旋风李逵','xuanfeng@qq.com',1), 
('00835','菩提老祖',null,1), 
('00391','白素贞',null,1), 
('00031','许仙','xuxian@qq.com',1), 
('00054','不想毕业',null,1), 
('51234','好好说话','say@qq.com',2), 
('83223','tellme',null,2), 
('09527','老外学中文','foreigner@qq.com',2); 

drop table if exists course; 
create table course( 
    id int primary key auto_increment, 
    name varchar(20) 
  );
insert into course(name) values 
('Java'),
('中国传统文化'),
('计算机原理'),
('语文'),
('高 阶数学'),
('英文');

drop table if exists score; 
create table score( 
      id int primary key auto_increment, 
      score DECIMAL, 
      student_id int, 
      course_id int 
   );
insert into score(score, student_id, course_id) values 
-- 黑旋风李逵 
(70.5, 1, 1),(98.5, 1, 3),(33, 1, 5),(98, 1, 6), 
-- 菩提老祖 
(60, 2, 1),(59.5, 2, 5), 
-- 白素贞 
(33, 3, 1),(68, 3, 3),(99, 3, 5), 
-- 许仙 
(67, 4, 1),(23, 4, 3),(56, 4, 5),(72, 4, 6), 
-- 不想毕业 
(81, 5, 1),(37, 5, 5),
-- 好好说话 
(56, 6, 2),(43, 6, 4),(79, 6, 6), 
-- tellme 
(80, 7, 2),(92, 7, 6);

内连接

语法:
select 字段 from 表1 别名1 [inner] join 表2 别名2 on 连 接条件 and 其他条件;
select 字段 from 表1 别名1,表2 别名2 where 连接条件 and 其他条件;
建议:多张表联合查询的时候,去找每个表之间的关系。
1、inner可以省略
2、或者可以省略join on
查询许仙同学的成绩
一:

select stu.name,sc.score,co.name 
from student stu ,score sc,course co 
where stu.id=sc.student_id 
and co.id=sc.course_id 
and stu.name='许仙';

在这里插入图片描述
二:

select stu.sn,stu.name,sco.score,cou.name as 课程名 
from student stu 
join score sco 
on stu.id = sco.student_id  
join course cou 
on sco.course_id = cou.id 
and stu.name='许仙';

在这里插入图片描述
总结:内连接 就是求几张表的 交集
在这里插入图片描述
** 查询每个同学的总成绩,及同学的个人信息不用求课程名**

select stu.sn, stu.name,sum(score)
from student stu,score sco
where stu.id=sco.student_id
group by stu.id;

在这里插入图片描述
查询所有同学的成绩,及同学的个人信息:

select stu.id,stu.sn,stu.name,sco.score,sco.course_id,co.name 
from student stu,score sco,course co 
where stu.id=sco.student_id 
and sco.course_id=co.id 
order by stu.id;

SELECT
 stu.id,
 stu.sn,
 stu.NAME,
 stu.qq_mail,
 sco.score,
 sco.course_id,
 cou.NAME
FROM
 student stu
 JOIN score sco ON stu.id = sco.student_id
 JOIN course cou ON sco.course_id = cou.id
ORDER BY stu.id;

在这里插入图片描述

外连接:左外连接 和 右外连接

左外连接

在进行连接查询的时候,以左边的表为准
在这里插入图片描述
查询每个同学的成绩,及同学的个人信息,如果该同学没有成绩,也需要显示该同学

 select stu.sn,stu.name,sco.score
 from student stu left join score sco
 on stu.id=sco.student_id
 group by stu.id;

在这里插入图片描述

右外连接

在这里插入图片描述
如果实现上述同样的功能,那么只需要这样修改就好了!!

select stu.id,stu.name,sco.score 
from score sco right join student stu
on stu.id = sco.student_id 
group by stu.id;

自连接

把一张表 看做两张表来使用。
显示所有“计算机原理”成绩比“Java”成绩高的成绩信息

– 先查询“计算机原理”和“Java”课程的id
select id,name from course where name=‘Java’ or name=‘计算机原理’;
在这里插入图片描述

– 再查询成绩表中,“计算机原理”成绩比“Java”成绩 好的信息
SELECT
s1.*
FROM
score s1,
score s2
WHERE
s1.student_id = s2.student_id
AND s1.score < s2.score
AND s1.course_id = 1
AND s2.course_id = 3;
在这里插入图片描述
方法二:使用join on

-- 也可以使用join on 语句来进行自连接查询
SELECT
 s1.* 
FROM
 score s1
 JOIN score s2 ON s1.student_id = s2.student_id
 AND s1.score < s2.score
 AND s1.course_id = 1
 AND s2.course_id = 3;

在这里插入图片描述
以上查询只显示了成绩信息,并且是分布执行的。要显示学生及成绩信息,并在一条语句显示:


SELECT
 stu.*,
 s1.score Java, 
 s2.score 计算机原理
FROM
 score s1
 JOIN score s2 ON s1.student_id = s2.student_id
 JOIN student stu ON s1.student_id = stu.id
 JOIN course c1 ON s1.course_id = c1.id
 JOIN course c2 ON s2.course_id = c2.id
 AND s1.score < s2.score
 AND c1.NAME = 'Java'
 AND c2.NAME = '计算机原理';

子查询

子查询是指嵌入在其他sql语句中的select语句,也叫嵌套语句

单行子查询:返回一行记录的子查询

例:查询与“不想毕业”同学一班的同班同学

 select * from student
  where classes_id
  =(select classes_id from student 、
  where name='不想毕业');

在这里插入图片描述

多行查询:返回多行记录的子查询

查询“语文”或“英文”课程的成绩信息

[NOT] IN关键字
-- 使用IN
select * from score where course_id in (select id from course where
name='语文' or name='英文');
-- 使用 NOT IN
select * from score where course_id not in (select id from course where
name!='语文' and name!='英文');

可以使用多列包含:

---插入重复的分数:score,student_id,course_id 列重复
insert into score(score, student_id, course_id) values
-- 黑旋风李逵
(70.5, 1, 1),(98.5, 1, 3),
-- 菩提老祖
(60, 2, 1);
---查询重复的分数
SELECT
 * 
FROM
 score 
WHERE
 ( score, student_id, course_id ) IN ( SELECT score, student_id, 
course_id FROM score GROUP BY score, student_id, course_id HAVING
count( 0 ) > 1 );
[NOT]EXISTS关键字

EXISTS(表达式) 只要这个表达式为真 就返回true

例如这条SQL语句:SELECT * FROM A WHERE EXISTS (SELECT 1 FROM B WHERE B.id = A.id);
1、首先执行一次外部查询,并缓存结果集,如 SELECT * FROM A
2、遍历外部查询结果集的每一行记录R,代入子查询中作为条件进行查询,如 SELECT 1 FROM B WHERE B.id = A.id
3、如果子查询有返回结果,则EXISTS子句返回TRUE,这一行R可作为外部查询的结果行,否则不能作为结果

-- 使用 EXISTS
select * from score sco where exists (select sco.id from course cou 
where (name='语文' or name='英文') and cou.id = sco.course_id);
-- 使用 NOT EXISTS
select * from score sco where not exists (select sco.id from course cou 
where (name!='语文' and name!='英文') and cou.id = sco.course_id);
  • 在from子句中使用子查询:子查询语句出现在from子句中。这里要用到数据查询的技巧,将子查询当作一个临时表使用。
    查询所有比中文系2019级3班平均分高的成绩信息
SELECT
 * 
FROM
 score sco,
 (
 SELECT
 avg( sco.score ) score 
 FROM
 score sco
 JOIN student stu ON sco.student_id = stu.id
 JOIN classes cls ON stu.classes_id = cls.id
 WHERE
 cls.NAME = '中文系2019级3班'
 ) tmp 
WHERE
 sco.score > tmp.score;

合并查询

为了合并多个select的执行结果,可以使用集合操作符union,union all.使用UNION和UNION ALL时,前后查询的结果集中,字段需要一致。

unoin

该操作符用于取得两个结果集的并集。当使用该操作符时,会自动去掉结果集中的重复行。
查询id<3,或者名字为“英文”的课程

select * from course where id<3
unoin 
select * from course where name='英文';

---或者使用or来实现
select * from course where id<3 or name='英文';

unoin all

该操作符用于取得两个结果集的并集,当使用该操作符时,不会去掉结果集中的重复行。
查询id<3,或者名字为“Java”的课程

select * from course where id<3
unoin all
select * from course where name='java';
  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值