MySQL联表查询基本操作之left-join常见的坑
作者:当我遇上你 时间:2024-01-14 20:49:22
概述
对于中小体量的项目而言,联表查询是再常见不过的操作了,尤其是在做报表的时候。然而校对数据的时候,您发现坑了吗?本篇文章就 mysql 常用联表查询复现常见的坑。
基础环境
建表语句
DROP TABLE IF EXISTS `role`;
CREATE TABLE `role` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`role_name` VARCHAR(50) DEFAULT NULL COMMENT '角色名',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='角色表';
insert into `role` VALUES(1, '管理员');
insert into `role` VALUES(2, '总经理');
insert into `role` VALUES(3, '科长');
insert into `role` VALUES(4, '组长');
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`role_id` int(11) NOT NULL COMMENT '角色id',
`user_name` VARCHAR(50) DEFAULT NULL COMMENT '用户名',
`sex` int(1) DEFAULT 0 COMMENT '性别',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用户表';
insert into `user` VALUES(1, 1, 'admin', 1);
insert into `user` VALUES(2, 2, '王经理', 1);
insert into `user` VALUES(3, 2, '李经理', 2);
insert into `user` VALUES(4, 2, '张经理', 2);
insert into `user` VALUES(5, 3, '王科长', 1);
insert into `user` VALUES(6, 3, '李科长', 1);
insert into `user` VALUES(7, 3, '吕科长', 2);
insert into `user` VALUES(8, 3, '邢科长', 1);
insert into `user` VALUES(9, 4, '范组长', 2);
insert into `user` VALUES(10, 4, '赵组长', 2);
insert into `user` VALUES(11, 4, '姬组长', 1);
数据如下
mysql> select * from role;
+----+-----------+
| id | role_name |
+----+-----------+
| 1 | 管理员 |
| 2 | 总经理 |
| 3 | 科长 |
| 4 | 组长 |
+----+-----------+
4 rows in set (0.00 sec)
mysql> select * from user;
+----+---------+-----------+------+
| id | role_id | user_name | sex |
+----+---------+-----------+------+
| 1 | 1 | admin | 1 |
| 2 | 2 | 王经理 | 1 |
| 3 | 2 | 李经理 | 2 |
| 4 | 2 | 张经理 | 2 |
| 5 | 3 | 王科长 | 1 |
| 6 | 3 | 李科长 | 1 |
| 7 | 3 | 吕科长 | 2 |
| 8 | 3 | 邢科长 | 1 |
| 9 | 4 | 范组长 | 2 |
| 10 | 4 | 赵组长 | 2 |
| 11 | 4 | 姬组长 | 1 |
+----+---------+-----------+------+
11 rows in set (0.00 sec)
基本业务
简单信息报表: 查询用户信息
mysql> SELECT
-> id,
-> user_name AS '姓名',
-> ( CASE WHEN sex = 1 THEN '男' WHEN sex = 2 THEN '女' ELSE '未知' END ) AS '性别'
-> FROM
-> USER;
+----+-----------+--------+
| id | 姓名 | 性别 |
+----+-----------+--------+
| 1 | admin | 男 |
| 2 | 王经理 | 男 |
| 3 | 李经理 | 女 |
| 4 | 张经理 | 女 |
| 5 | 王科长 | 男 |
| 6 | 李科长 | 男 |
| 7 | 吕科长 | 女 |
| 8 | 邢科长 | 男 |
| 9 | 范组长 | 女 |
| 10 | 赵组长 | 女 |
| 11 | 姬组长 | 男 |
+----+-----------+--------+
查询每个角色名称及对应人员中女性数量
mysql> SELECT
-> r.id,
-> r.role_name AS role,
-> count( u.sex ) AS sex
-> FROM
-> role r
-> LEFT JOIN USER u ON r.id = u.role_id
-> AND u.sex = 2
-> GROUP BY
-> r.role_name
-> ORDER BY
-> r.id ASC;
+----+-----------+-----+
| id | role | sex |
+----+-----------+-----+
| 1 | 管理员 | 0 |
| 2 | 总经理 | 2 |
| 3 | 科长 | 1 |
| 4 | 组长 | 2 |
+----+-----------+-----+
4 rows in set (0.00 sec)
假如我们把性别过滤的条件改为 where 操作结果会怎么样呢?
mysql> SELECT
-> r.id,
-> r.role_name AS role,
-> count( u.sex ) AS sex
-> FROM
-> role r
-> LEFT JOIN USER u ON r.id = u.role_id
-> WHERE
-> u.sex = 2
-> GROUP BY
-> r.role_name
-> ORDER BY
-> r.id ASC;
+----+-----------+-----+
| id | role | sex |
+----+-----------+-----+
| 2 | 总经理 | 2 |
| 3 | 科长 | 1 |
| 4 | 组长 | 2 |
+----+-----------+-----+
3 rows in set (0.00 sec)
这里可以看到角色数据不完整了。
找出角色为总经理的员工数量
mysql> SELECT
-> r.id,
-> r.role_name AS role,
-> count( u.sex ) AS sex
-> FROM
-> role r
-> LEFT JOIN USER u ON r.id = u.role_id
-> WHERE
-> r.role_name = '总经理'
-> GROUP BY
-> r.role_name
-> ORDER BY
-> r.id ASC;
+----+-----------+-----+
| id | role | sex |
+----+-----------+-----+
| 2 | 总经理 | 3 |
+----+-----------+-----+
1 row in set (0.00 sec)
同样将过滤条件由 where 改为 on
mysql> SELECT
-> r.id,
-> r.role_name AS role,
-> count( u.sex ) AS sex
-> FROM
-> role r
-> LEFT JOIN USER u ON r.id = u.role_id
-> AND r.role_name = '总经理'
-> GROUP BY
-> r.role_name
-> ORDER BY
-> r.id ASC;
+----+-----------+-----+
| id | role | sex |
+----+-----------+-----+
| 1 | 管理员 | 0 |
| 2 | 总经理 | 3 |
| 3 | 科长 | 0 |
| 4 | 组长 | 0 |
+----+-----------+-----+
4 rows in set (0.00 sec)
这里可以看到数据多余了
总结
在 left join 语句中,左表过滤必须放 where 条件中,右表过滤必须放 on 条件中,这样结果才能不多不少,刚刚好。
来源:https://juejin.im/post/5ebd5a8b51882573a924e2fd
标签:联表查询,left,join
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
python数据分析之单因素分析线性拟合及地理编码
2021-02-09 06:46:20
![](https://img.aspxhome.com/file/2023/6/65516_0s.png)
Golang汇编之控制流深入分析讲解
2024-05-08 10:15:11
如何优化下面这段代码?
2010-01-23 11:30:00
JavaScript 中级笔记 第四章 闭包
2024-04-18 10:53:09
使用Python实现画一个中国地图
2023-10-01 00:39:36
![](https://img.aspxhome.com/file/2023/0/131140_0s.jpg)
SQL Server 2000“设备激活错误”的解决方法
2024-01-12 15:30:23
![](https://img.aspxhome.com/file/2023/3/94443_0s.jpg)
一空间多域名绑定不同目录方法
2009-03-09 18:32:00
Python实现微信消息防撤回功能的实例代码
2023-10-05 14:20:51
![](https://img.aspxhome.com/file/2023/9/101029_0s.gif)
Python中 传递值 和 传递引用 的区别解析
2023-12-26 07:05:22
![](https://img.aspxhome.com/file/2023/0/123800_0s.jpg)
Python模块/包/库安装的六种方法及区别
2021-11-03 15:53:56
![](https://img.aspxhome.com/file/2023/4/72964_0s.png)
Django框架模板语言实例小结【变量,标签,过滤器,继承,html转义】
2023-11-11 06:41:39
解决MYSQL出现Can't create/write to file '/tmp/#sql_5c0_0.MYD'的问题
2024-01-16 07:53:39
Python3实现英文字母转换哥特式字体实例代码
2023-08-04 10:28:11
![](https://img.aspxhome.com/file/2023/0/98880_0s.jpg)
python基础教程之自定义函数介绍
2021-09-12 13:10:24
Linux下安装Mysql多实例作为数据备份服务器实现多主到一从多实例的备份
2024-01-13 19:12:27
解决matplotlib库show()方法不显示图片的问题
2021-08-25 19:53:34
ASP JSON类源码
2011-04-30 16:38:00
Python爬虫实战之用selenium爬取某旅游网站
2021-03-25 10:28:36
![](https://img.aspxhome.com/file/2023/4/72724_0s.png)
python能否java成为主流语言吗
2022-10-16 18:34:11
功能强大,代码简单的管理菜单
2008-07-11 16:52:00