概述
对于中小体量的项目而言,联表查询是再常见不过的操作了,尤其是在做报表的时候。然而校对数据的时候,您发现坑了吗?本篇文章就 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 条件中,这样结果才能不多不少,刚刚好。
到此这篇关于MySQL联表查询基本操作之left-join常见坑的文章就介绍到这了,更多相关MySQL联表查询left-join内容请搜索小牛知识库以前的文章或继续浏览下面的相关文章希望大家以后多多支持小牛知识库!
问题内容: 意思是我想根据from的值动态离开联接表,因为从那里取了。 但是上面的查询结果 表table_name不存在 我的mysql限制错误? 问题答案: 表名和列名在SQL查询中不能是动态的。因此,您必须使用2个查询或存储过程以编程方式应用逻辑,请参见此处的示例:http : //forums.mysql.com/read.php?98,126506,126598#msg-126598
问题内容: 询问: 两个表都有8k记录,但是为什么它却很慢,需要2-3分钟,有时还要多一些? OMG,该查询使mysql服务器停机。将在一秒钟内回复你们人民:( 建议所有为这些索引编制索引的人都是正确的。是的,我写的查询既傻又马车。感谢纠正我。 问题答案: 还考虑对表建立索引。我们正在100万以上的记录表上运行多个左联接,这些联接不需要一两秒钟就能返回结果。
本文向大家介绍mysql中各种常见join连表查询实例总结,包括了mysql中各种常见join连表查询实例总结的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了mysql中各种常见join连表查询。分享给大家供大家参考,具体如下: 通常我们需要连接多个表查询数据,以获取想要的结果。 一、连接可以分为三类: (1) 内连接:join,inner join (2) 外连接:left jo
本文向大家介绍mysql学习笔记之表的基本操作,包括了mysql学习笔记之表的基本操作的使用技巧和注意事项,需要的朋友参考一下 创建表 create table 表名 create table if not exists 表名 显示当前库下的所有表 show tables; 查看表的结构 describe 表名 简写 desc 表名 查看表的详细 show create table 表名 删除表
问题内容: 我有一张表,每个日期都有一些统计信息,我想用MySQL列出。对于某些日期,将没有统计信息,因此结果应类似于以下内容: 2013-03-01:3 2013-03-02:2 2013-03-03:0 2013-03-04:1 我发现可以用一个单独的表(包含所有可能的日期和LEFT JOIN)来解决用0-零-填补空白的问题。到目前为止,一切都很好。 统计信息(展示次数)位于“ campaig
问题内容: 可以说我有一个: 并带有第一个外键的: 现在,我想将其作为查询结果: 我该如何实现? 我已经尝试过: 但结果似乎忽略了空值: 求助:感谢Mahmoud Gamal:(加上GROUP BY)用此查询解决了 问题答案: 您必须使用以获得第一个表与第二个表的所有可能组合。然后将这些与第二张表组合。就像是: