mysql-实战案例 (超详细版)

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: mysql-实战案例 (超详细版)

🚀多表查询综合案例

数据环境准备:

create table salgrade(
grade int, --工资等级
lowsal int, --这个等级的最低工资
hightsal int --这个等级的最高工资
) comment '工资等级表'; 
insert into salgrade values (1,0,3000);
insert into salgrade values (2,3001,5000);
insert into salgrade values (3,5001,8000);
insert into salgrade values (4,8001,10000);
insert into salgrade values (5,10001,15000);
insert into salgrade values (6,15001,20000);
insert into salgrade values (7,20001,25000);
insert into salgrade values (8,25001,30000);

在这个案例中,我们主要运用上面所讲解的多表查询的语法,完成以下的12个需求即可,而这里主要涉及到的表就三张:empcp员工表、dept部门表、salgrade工资等级表 。

✨查询员工的姓名、年龄、职位、部门信息 (隐式内连接,需要消除笛卡尔积)

需要用的表 : empcp , dept

需要的连接条件: empcp.dept_id = dept.id

这里再重复一下,如果给表起了别名你就必须要用别名去定义字段了

select e.name , e.age , e.job , d.name from empcp e , dept d where e.dept_id = d.id;

执行:


✨查询年龄小于29岁的员工的姓名、年龄、职位、部门信息(显式内连接)

需要用的表 : empcp , dept

需要的连接条件: empcp.dept_id = dept.id

在DQL语句中我们的查询条件在哪个关键字之后?是在where之后

select e.name , e.age , e.job , d.name from empcp e inner join dept d on e.dept_id =
d.id where e.age < 29;

执行:

✨查询拥有员工的部门ID、部门名称 (难点在于有一个部门是没有员工的)

需要用的表 : empcp , dept

需要的连接条件: empcp.dept_id = dept.id

换个思路:我们要去查拥有员工的部门,其实就是查询部门表和员工表之间的交集,怎么去查询两个表之间的交集,其实实际就是用内连接

select  * from empcp e , dept d where e.dept_id = d.id ;

而我们只要返回部门id,和部门名称,所以把*改为d.id,d.name

发现重复,去重,用关键字distinct

select distinct d.id , d.name from empcp e , dept d where e.dept_id = d.id;

执行:


✨查询所有年龄大于41岁的员工, 及其归属的部门名称; 如果员工没有分配部门, 也需要展示出来

(因为没有部门的也要展示,只能使用外连接,外连接大部分采用左外,因为右外也可以通过调换表顺序改为左外)

需要用的表 : empcp , dept

需要的连接条件: empcp.dept_id = dept.id

思路:

select * from empcp e left join dept d on e.dept_id = d.id where e.age > 40;

然后把*换成员工表的所有字段e.*,和部门名称d.name,如下

select e.*, d.name from empcp e left join dept d on e.dept_id = d.id where e.age >
41 ;

执行:


✨查询所有员工的工资等级 (发现跟部门没有关系,而员工表中只有工资,没有等级,所以涉及新表salgrade)

需要用的表 : empcp , salgrade

需要的连接条件 : empcp.salary >= salgrade.lowsal and empcp.salary <= salgrade.hightsal

1.empcp表与salgrade表进行联合查询

2.empcp表与salgrade之间有没有外键关联?没有,如果要说empcp和salgrade之间没有外键关联,怎么产生关系呢?它们之间产生关系实际上通过一个字段salary产生关系,我们要去判断员工薪资等级,就要让这个薪资介于最小值与最大值之间,这就是连接条件


3.select *  from empcp e,salgrade s where e.salary >= s.lowsal and e.salary <= s.hightsal ;

再改细节,*变为empcp表的全部字段e*,和salgrade表中的grade字段s.grade

-- 方式一
select e.* , s.grade , s.lowsal, s.hightsal from empcp e , salgrade s where e.salary >= s.lowsal and e.salary <= s.hightsal;
-- 方式二
select e.* , s.grade , s.lowsal, s.hightsal from empcp e , salgrade s where e.salary between s.lowsal and s.hightsal;

执行:

✨查询 "研发部" 所有员工的信息及工资等级


需要用的表 : empcp , salgrade , dept

需要的连接条件 : empcp.salary between salgrade.lowsal and salgrade.hightsal , dept.id= empcp.dept_id

查询条件 : dept.name = '研发部'

梳理:

1.因为员工表中只有部门的id,而这里要根据部门的名词去查,所以这个里面涉及的表结构将会是三张表

2.那么可想而知,三张表我们连接连接的条件会有几个?2个

3.联查n张表,至少需要n-1个条件

4.我们去梳理连接条件的时候,不要一股脑的梳理三张表的条件,两个两个去梳理

5.首先empcp和salgrade的连接条件我们已经梳理过了,就是empcp.salary between salgrade.lowsal and salgrade.hightsal

6.再梳理第二个条件empcp表和dept表它们什么条件,不就是empcp.dept_id = dept.id

代码思路:

1.select * from empcp e , dept d , salgrade s ; 这种我们称之为隐式内连接

2.连接的条件将在where之后编写

3.select * from empcp e , dept d , salgrade s where e.dept_id = d.id;

4.如果是多个连接条件之间使用and进行连接

5.select * from empcp e , dept d , salgrade s where e.dept_id = d.id and e.salary between s.lowsal and s.hightsal;

6为了让语法更加清晰,可以用括号完善一下

7.select * from empcp e , dept d , salgrade s where e.dept_id = d.id and (e.salary between s.lowsal and s.hightsal);

8.还有一个查询条件,记住如果说在连接查询中还有额外的查询条件,此时直接在where后继续跟着写就可以

9.select * from empcp e , dept d , salgrade s where e.dept_id = d.id and (e.salary between s.lowsal and s.hightsal) and d.name = '研发部';

10.最后将*改动,写为需要取的字段e.*员工信息,s.grade工资等级,如下:

select e.* , s.grade from empcp e , dept d , salgrade s where e.dept_id = d.id and (
e.salary between s.lowsal and s.hightsal ) and d.name = '研发部';
我们可以格式化一下代码更清楚
--
select e.*, s.grade
from empcp e,
     dept d,
     salgrade s
where e.dept_id = d.id
  and (
    e.salary between s.lowsal and s.hightsal)
  and d.name = '研发部';

执行:

✨查询 "研发部" 员工的平均工资

需要用的表 : empcp , dept

需要的连接条件  : empcp.dept_id = dept.id

要求平均数的聚合函数是avg

select * from empcp e , dept d where e.dept_id = d.id ;

还有一个查询条件 查询研发部 d.name = "研发部"

select * from empcp e , dept d where e.dept_id = d.id and d.name = "研发部"

基本的sql编写完了再对*进行优化,如下

select avg(e.salary) from empcp e, dept d where e.dept_id = d.id and d.name = '研发部';

执行:

✨查询工资比 "n" 高的员工信息。

思路:

拆解为两步

1.查询 "n" 的薪资

select salary from empcp where name = 'n';

2.查询比她工资高的员工数据

select * from empcp where salary > ( select salary from empcp where name = 'n' );
--后面其实就是刚刚查出来的14000,这个14000就是上面那条语句查询返回的结果,所以我们要通过子查询来实现,把上面的语句放到下面,然后再加上括号

执行:

执行:

✨查询比平均薪资高的员工信息

思路:

拆解为两步

1.查询员工的平均薪资

select avg(salary) from empcp;

2.查询比平均薪资高的员工信息

select * from empcp where salary > ( select avg(salary) from empcp );

执行:

执行:


✨查询低于本部门平均工资的员工信息

思路:

拆解为两步

1.查询指定部门平均薪资

比如查询1号部门的平均薪资

select * from emp e1 where e1.dept_id = 1

*改avg(e1.salary) 此时只能用别名e1

select avg(e1.salary) from empcp e1 where e1.dept_id = 1;
select avg(e1.salary) from empcp e1 where e1.dept_id = 2;

2.查询低于本部门平均工资的员工信息

思路:

1.首先查询出所有的员工数据

2.select * from empcp;

3.此时所有的员工数据都拿到了,在此基础上增加一个查询条件,salary<当前部门的平均薪资

4.当前部门的平均薪资,只需要把当前的部门薪资传递下来,也就是上面的那句sql语句

select * from empcp where salary < (select avg(e1.salary) from empcp e1 where e1.dept_id = 1);

5.但是里面写死的部门id我们要替换掉,比如当我们去判断第一行数据时,需要传递的部门id是5,去判断第二行数据时,需要传递的部门id是1

6.此时我们只需要把dept_id这个字段传进来即可,我们需要把empcp起一个别名e2

7.这里就应该是select * from empcp e2 where e2.salary < (select avg(e1.salary) from empcp e1 where e1.dept_id = 1);e2.salary < 当前部门的平均薪资,再把当前部门传进来

8.当前部门应该就是e2.dept_id,代码如下

select * from empcp e2 where e2.salary < ( select avg(e1.salary) from empcp e1 where
e1.dept_id = e2.dept_id );

执行:

执行:


为了验证查询到的字段,我们可以在*之后增加一个输出字段, 我们可以把子查询直接粘贴过来,这个不就是当前部门的平均薪资嘛,我们再取一个别名为”当前部门的平均薪资“

select e2.*, (select avg(e1.salary) from empcp e1 where e1.dept_id = e2.dept_id) '当前部门平均薪资'
from empcp e2
where e2.salary < (select avg(e1.salary) from empcp e1 where e1.dept_id = e2.dept_id);


对比后发现,是正确的

✨查询所有的部门信息, 并统计部门的员工人数

1.先查询所有部门信息

2.select id,name from dept;

3.此时查出了所有的部门id,以及部门的名称

4.接着考虑如何统计部门的员工数量

5.select count(*) from empcp where dept_id = 1;

6.这条语句是统计1号部门的员工数量

7.接下来需要考虑,在上面的语句下如何统计1号部门的员工数量,2号部门的员工数量

8.我再添加一个查询id,此时select id,name,id '人数' from dept;

9.这个id本身不是查询人数,我想要后面的dept.id查询为人数,能不能根据id查询出人数呢?可以,就是这句,统计1号部门的员工数量select count(*) from empcp where dept_id = 1;

10.变为select id,name,(select count(*) from empcp where dept_id = id) '人数' from dept;  加上括号,变成一个子查询存在

11.这里面最需要注意的是标红的dept_id是empcp表的id,紫id是dept表的id

12.此时就要给表起对应的别名,给empcp起别名为e,给dept起名为d

13.select d.id, d.name,(select count(*) from empcp e where e.dept_id =d.id) '人数' from dept  d;

select d.id, d.name , ( select count(*) from empcp e where e.dept_id = d.id ) '人数' from dept d;

执行:

✨查询所有学生的选课情况, 展示出学生名称, 学号, 课程名称


表 : student , course , student_course

连接条件: student.id = student_course.studentid , course.id = student_course.courseid

1.之前讲解表与表之间的关系时,特别说明了学生和课程的关系是多对多的关系,一个学生可以选择多个课程,一个课程也可以供多个学生选择,涉及的表有三张

2.这三张表会通过中间表student_course来维护它们之间的关系

3.三张表在消除笛卡尔积的时候需要3-1个关系

4.先梳理一下连接条件

5.student.id = student_course.studentid,course.id = student_course.courseid

ps:第一张是student表,第二张是student_course表,第三张是course表

6.select * from student s,student_course sc,course c where s.id = sc.studentid and  c.id = sc.courseid ;

7.再对*优化

8.我们只需要展示学生的姓名, 学号,课程也就是s.name , s.no , c.name ,代码如下

select s.name , s.no , c.name from student s , student_course sc , course c where
s.id = sc.studentid and sc.courseid = c.id ;

执行:

相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。 &nbsp; 相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/mysql&nbsp;
相关文章
|
4月前
|
负载均衡 算法 关系型数据库
大数据大厂之MySQL数据库课程设计:揭秘MySQL集群架构负载均衡核心算法:从理论到Java代码实战,让你的数据库性能飙升!
本文聚焦 MySQL 集群架构中的负载均衡算法,阐述其重要性。详细介绍轮询、加权轮询、最少连接、加权最少连接、随机、源地址哈希等常用算法,分析各自优缺点及适用场景。并提供 Java 语言代码实现示例,助力直观理解。文章结构清晰,语言通俗易懂,对理解和应用负载均衡算法具有实用价值和参考价值。
大数据大厂之MySQL数据库课程设计:揭秘MySQL集群架构负载均衡核心算法:从理论到Java代码实战,让你的数据库性能飙升!
|
10月前
|
缓存 关系型数据库 MySQL
MySQL索引策略与查询性能调优实战
在实际应用中,需要根据具体的业务需求和查询模式,综合运用索引策略和查询性能调优方法,不断地测试和优化,以提高MySQL数据库的查询性能。
556 66
|
5月前
|
关系型数据库 MySQL 大数据
大数据新视界--大数据大厂之MySQL 数据库课程设计:MySQL 数据库 SQL 语句调优的进阶策略与实际案例(2-2)
本文延续前篇,深入探讨 MySQL 数据库 SQL 语句调优进阶策略。包括优化索引使用,介绍多种索引类型及避免索引失效等;调整数据库参数,如缓冲池、连接数和日志参数;还有分区表、垂直拆分等其他优化方法。通过实际案例分析展示调优效果。回顾与数据库课程设计相关文章,强调全面认识 MySQL 数据库重要性。为读者提供综合调优指导,确保数据库高效运行。
|
10月前
|
前端开发 关系型数据库 MySQL
PHP与MySQL动态网站开发实战指南####
【10月更文挑战第21天】 本文将深入浅出地探讨如何使用PHP与MySQL构建一个动态网站,从环境搭建到项目部署,全程实战演示。无论你是编程新手还是希望巩固Web开发技能的老手,都能在这篇文章中找到实用的技巧和启发。我们将一起探索如何通过PHP处理用户请求,利用MySQL存储数据,并最终呈现动态内容给用户,打造属于自己的在线平台。 ####
389 0
|
7月前
|
存储 SQL 关系型数据库
服务器数据恢复—云服务器上mysql数据库数据恢复案例
某ECS网站服务器,linux操作系统+mysql数据库。mysql数据库采用innodb作为默认存储引擎。 在执行数据库版本更新测试时,操作人员误误将在本来应该在测试库执行的sql脚本在生产库上执行,导致生产库上部分表被truncate,还有部分表中少量数据被delete。
172 25
|
8月前
|
SQL 存储 关系型数据库
MySQL秘籍之索引与查询优化实战指南
最左前缀原则。不冗余原则。最大选择性原则。所谓前缀索引,说白了就是对文本的前几个字符建立索引(具体是几个字符在建立索引时去指定),比如以产品名称的前 10 位来建索引,这样建立起来的索引更小,查询效率更快!
281 22
 MySQL秘籍之索引与查询优化实战指南
|
7月前
|
SQL 关系型数据库 MySQL
数据库数据恢复——MySQL简介和数据恢复案例
MySQL数据库数据恢复环境&故障: 本地服务器,安装的windows server操作系统。 操作系统上部署MySQL单实例,引擎类型为innodb,表空间类型为独立表空间。该MySQL数据库没有备份,未开启binlog。 人为误操作,在用Delete命令删除数据时未添加where子句进行筛选导致全表数据被删除,删除后未对该表进行任何操作。
|
9月前
|
存储 关系型数据库 MySQL
10个案例告诉你mysql不使用子查询的原因
大家好,我是V哥。上周与朋友讨论数据库子查询问题,深受启发。为此,我整理了10个案例,详细说明如何通过优化子查询提升MySQL性能。主要问题包括性能瓶颈、索引失效、查询优化器复杂度及数据传输开销等。解决方案涵盖使用EXISTS、JOIN、IN操作符、窗口函数、临时表及索引优化等。希望通过这些案例,帮助大家在实际开发中选择更高效的查询方式,提升系统性能。关注V哥,一起探讨技术,欢迎点赞支持!
472 5
|
9月前
|
关系型数据库 MySQL 数据库
数据库数据恢复—MYSQL数据库文件损坏的数据恢复案例
mysql数据库文件ibdata1、MYI、MYD损坏。 故障表现:1、数据库无法进行查询等操作;2、使用mysqlcheck和myisamchk无法修复数据库。
|
10月前
|
安全 关系型数据库 MySQL
PHP与MySQL动态网站开发实战指南####
——深入探索LAMP栈下的高效数据交互与处理技巧 ####

推荐镜像

更多