你真的懂SQL的连接查询吗?Mysql Join连接查询一锅端

你真的懂SQL的连接查询吗?Mysql Join连接查询一锅端

SQL-增删改查一锅端中,我详细罗列了基本SQL的写法,对其中的Join连接查询也作了一些应用举例,但你真的理解这个Join连接,并熟练使用了么?

如果你对Join连接查询一知半解,恭喜你,捡到宝了,废话不多说,直接开干,大家先把下面这张表,在你的Mysql中建好,等下要用

当然你也可以选择不建,随便看看

CREATE TABLE `t_dept` ( 
`id` INT(11) NOT NULL AUTO_INCREMENT, 
`deptName` VARCHAR(30) DEFAULT NULL, 
`address` VARCHAR(40) DEFAULT NULL, 
PRIMARY KEY (`id`) 
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8; 
CREATE TABLE `t_emp` ( 
`id` INT(11) NOT NULL AUTO_INCREMENT, 
`name` VARCHAR(20) DEFAULT NULL, 
`age` INT(3) DEFAULT NULL, 
`deptId` INT(11) DEFAULT NULL, 
empno int not null, 
PRIMARY KEY (`id`), 
KEY `idx_dept_id` (`deptId`) 
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8; 
INSERT INTO t_dept(deptName,address) VALUES('华山','华山'); 
INSERT INTO t_dept(deptName,address) VALUES('丐帮','洛阳'); 
INSERT INTO t_dept(deptName,address) VALUES('峨眉','峨眉山'); 
INSERT INTO t_dept(deptName,address) VALUES('武当','武当山'); 
INSERT INTO t_dept(deptName,address) VALUES('明教','光明顶'); 
INSERT INTO t_dept(deptName,address) VALUES('少林','少林寺'); 
INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('风清扬',90,1,100001);
INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('岳不群',50,1,100002); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('令狐冲',24,1,100003); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('洪七公',70,2,100004); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('乔峰',35,2,100005); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('灭绝师太',70,3,100006); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('周芷若',20,3,100007); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('张三丰',100,4,100008); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('张无忌',25,5,100009); INSERT INTO t_emp(NAME,age,deptId,empno) VALUES('韦小宝',18,null,100010);

所有连接查询基于下面这张图

这是一张用韦恩图表示的双表连接查询,如图所示,两张表,你要哪部分数据,你就对应选择相关的连接查询即可,在下面的示例中,你应该与本图对照来学习,如此你便能理所当然,掌握所有相关的连接查询

  • 查询所有有门派人员的信息(要求显示门派名称)

对应的韦恩图:

SELECT e.name,d.deptNameFROM t_emp e INNER JOIN t_dept d ON e.deptId=d.id;

  • 列出所有人员及其门派信息

分析:要求所有人员,包括无门派人员,应该使用完全版左连接或右连接,这里就不在上图了,你应该回去看看那个图(提示,图示第一个)

SELECT e.name,d.deptName FROM t_emp e LEFT JOIN t_dept d ON e.deptId=d.id ;

  • 查询所有无人门派

分析:要求所有门派,还是左连接或右连接,不过得是阉割版的左连接,图示第三个

SELECT d.* FROM t_dept d LEFT JOIN t_emp e ON e.deptId=d.id WHERE e.deptId is NULL ;

  • 查询所有人员和门派的对应关系
SELECT * FROM t_emp e LEFT JOIN t_dept d ON e.`deptId`=d.`id` 
UNION 
SELECT * FROM t_emp e RIGHT JOIN t_dept d ON e.`deptId`=d.`id`;

分析:就是全查呗,一个左连加一个右连搞定

  • 所有没有入门派的人员和没人入的门派
SELECT * FROM t_emp e LEFT JOIN t_dept d ON e.`deptId`=d.`id` WHERE e.deptId IS NULL 
UNION 
SELECT * FROM t_dept d LEFT JOIN t_emp e ON d.`id`=e.`deptId` WHERE e.`deptId` IS NULL;

分析:就是取韦恩图 各自独立的部门,那么就把两边独占的那部分查出来,再联合起来,就可以了呀,细心的朋友可能会发现,其实用最后图示最后一个例子(FUll join)不也一样可以嘛,甚至更轻便,是的,但是需要说明的是

Mysql不支持Full Join(全关联)查询

接下来增加一点小难度哈,我们对部门表添加CEO字段,代表各门派掌门人

ALTER TABLE `t_dept` add CEO INT(11) ;
update t_dept set CEO=2 where id=1; 
update t_dept set CEO=4 where id=2;
update t_dept set CEO=6 where id=3; 
update t_dept set CEO=8 where id=4; 
update t_dept set CEO=9 where id=5;
  • 求各个门派对应的掌门人名称

SELECT d.deptName,e.name FROM t_dept d LEFT JOIN t_emp e ON d.ceo=e.id

分析:所有门派–>完全版左连接解决

  • 求所有掌门人平均年龄

SELECT AVG(e.age) FROM t_dept d LEFT JOIN t_emp e ON d.ceo=e.id

分析:同上

以上,是不是觉得连接来连接去的,不就这点东西了呢?
最后:Join连接查询比where查询并没有性能上的优越性,但它所代表的交叉连接明显更具有数学意义,用好join能让你的sql语句写得更加规范!

发布了182 篇原创文章 · 获赞 121 · 访问量 5万+

猜你喜欢

转载自blog.csdn.net/JunSIrhl/article/details/105623587