【如何成为SQL高手】第九关:高级复杂查询

简介: 【如何成为SQL高手】第九关:高级复杂查询

环境准备,通过以下建表


-- 创建数据库
CREATE DATABASE mes_db charset utf8mb4;
-- 切换数据库
USE mes_db;
-- 创建学生表
CREATE TABLE student (
no VARCHAR(20) PRIMARY KEY,
name VARCHAR(20) NOT NULL,
sex VARCHAR(10) NOT NULL,
birthday DATE, -- 生日
class VARCHAR(20) -- 所在班级
);
-- 创建教师表
CREATE TABLE teacher (
no VARCHAR(20) PRIMARY KEY,
name VARCHAR(20) NOT NULL,
sex VARCHAR(10) NOT NULL,
birthday DATE,
profession VARCHAR(20) NOT NULL, -- 职称
department VARCHAR(20) NOT NULL -- 部门
);
-- 创建课程表
CREATE TABLE course (
no VARCHAR(20) PRIMARY KEY,
name VARCHAR(20) NOT NULL,
t_no VARCHAR(20) NOT NULL, -- 教师编号
-- 表示该 t_no 来自于 teacher 表中的 no 字段值
FOREIGN KEY(t_no) REFERENCES teacher(no)
);
-- 成绩表
CREATE TABLE score (
s_no VARCHAR(20) NOT NULL, -- 学生编号
c_no VARCHAR(20) NOT NULL, -- 课程号
degree DECIMAL, -- 成绩
-- 表示该 s_no, c_no 分别来自于 student, course 表中的 no 字段值
FOREIGN KEY(s_no) REFERENCES student(no),
FOREIGN KEY(c_no) REFERENCES course(no),
-- 设置 s_no, c_no 为联合主键
PRIMARY KEY(s_no, c_no)
);
-- 查看所有表
SHOW TABLES;
-- 添加学生表数据
INSERT INTO student VALUES('101', '曾华', '男', '1977-09-01', '95033');
INSERT INTO student VALUES('102', '匡明', '男', '1975-10-02', '95031');
INSERT INTO student VALUES('103', '王丽', '女', '1976-01-23', '95033');
INSERT INTO student VALUES('104', '李军', '男', '1976-02-20', '95033');
INSERT INTO student VALUES('105', '王芳', '女', '1975-02-10', '95031');
INSERT INTO student VALUES('106', '陆军', '男', '1974-06-03', '95031');
INSERT INTO student VALUES('107', '王飘飘', '男', '1976-02-20', '95033');
INSERT INTO student VALUES('108', '张全蛋', '男', '1975-02-10', '95031');
INSERT INTO student VALUES('109', '赵铁柱', '男', '1974-06-03', '95031');
-- 添加教师表数据
INSERT INTO teacher VALUES('804', '李诚', '男', '1958-12-02', '副教授', '计算机系');
INSERT INTO teacher VALUES('856', '张旭', '男', '1969-03-12', '讲师', '电子工程系');
INSERT INTO teacher VALUES('825', '王萍', '女', '1972-05-05', '助教', '计算机系');
INSERT INTO teacher VALUES('831', '刘冰', '女', '1977-08-14', '助教', '电子工程系');
-- 添加课程表数据
INSERT INTO course VALUES('3-105', '计算机导论', '825');
INSERT INTO course VALUES('3-245', '操作系统', '804');
INSERT INTO course VALUES('6-166', '数字电路', '856');
INSERT INTO course VALUES('9-888', '高等数学', '831');
-- 添加添加成绩表数据
INSERT INTO score VALUES('103', '3-105', '92');
INSERT INTO score VALUES('103', '3-245', '86');
INSERT INTO score VALUES('103', '6-166', '85');
INSERT INTO score VALUES('105', '3-105', '88');
INSERT INTO score VALUES('105', '3-245', '75');
INSERT INTO score VALUES('105', '6-166', '79');
INSERT INTO score VALUES('109', '3-105', '76');
INSERT INTO score VALUES('109', '3-245', '68');
INSERT INTO score VALUES('109', '6-166', '81');


3e95ced999e34b799a94fdeb3af79db2.png

1.ANY用法



查询课程 3-105 且成绩至少高 3-245 的 score 表,DESC ( 降序 )


#ANY: 符合 SQL 语句中的任意条件

#也就是说,3-105成绩中,只要有一个大于从3-245筛选出来的任意行就符合条件

#最后根据降序查询结果

SELECT * FROM score WHERE c_no = ‘3-105’ AND degree > ANY(

SELECT degree FROM score WHERE c_no = ‘3-245’

) ORDER BY degree DESC;


efe6e5dd7c234d8cac0df58d14cda097.png

⛳️2.ALL的用法



查询课程 3-105 且成绩高于 3-245 的 score 表


#只需对上一道题稍作修改。

#ALL: 符合 SQL 语句中的所有条件。

#也就是说,在3-105成绩中,都要大于从3-245筛选出来全部行才算符合条件

SELECT * FROM score WHERE c_no = ‘3-105’ AND degree > ALL(

SELECT degree FROM score WHERE c_no = ‘3-245’);


70f20bf643394b1fa022e0a2457a6d6e.png


⛳️3. 自连接



查询某课程成绩比该课程平均成绩低的 score 表

#将表 b 作用于表 a 中查询数据

#score 别名a,b 如此就能用 a.c_no=b.c_no 作为条件执行查询

SELECT * FROM score a WHERE degree < (

(SELECT AVG(degree) FROM score b WHERE a.c_no = b.c_no));


271abb84435a4fbc914e5c9906de0d3f.png


⛳️4. 排序



😆 4.1 多字段排序


以 class 和 birthday 从大到小的顺序查询 student 表

SELECT * FROM student ORDER BY class DESC, birthday;

f502db9ab1684c00a652d6b54e1728d1.png


😆 4.2 等级处理


建立一个 grade 表代表学生的成绩等级,并插入数据:

CREATE TABLE grade (

low INT(3),

upp INT(3),

grade char(1)

);

INSERT INTO grade VALUES (90, 100, ‘A’);

INSERT INTO grade VALUES (80, 89, ‘B’);

INSERT INTO grade VALUES (70, 79, ‘C’);

INSERT INTO grade VALUES (60, 69, ‘D’);

INSERT INTO grade VALUES (0, 59, ‘E’);

SELECT * FROM grade;

–查询所有学生的 s_no 、c_no 和 grade 列

SELECT s_no, c_no, grade FROM score, grade

WHERE degree BETWEEN low AND upp;


1794df7102494bc3b3144fd2d459bce8.png

😆 4.3 排名


create table scores_tb (

id int auto_increment primary key,

xuehao int not null,

score int not null

) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;


insert into scores_tb (xuehao,score)

values (1001,89),

(1002,99),

(1003,96),

(1004,96),

(1005,92),

1006,90),

(1007,90),

(1008,94);

select * from scores_tb;


75da7b7214ca433ba63aa488b390069d.png


MySQL8.0 中可以利用 ROW_NUMBER(),DENSE_RANK(),RANK()

三个窗口函数实现上述三种排名


普通排名:

按分数高低直接排名,从 1 开始,往下排,类似于 row number

select xuehao,score,

ROW_NUMBER() OVER(order by score desc)

as row_r from scores_tb;


9d16650bf14540bb9064c19dbb7ccacf.png


数相同,名次相同,排名无间隔

select xuehao,score,

DENSE_RANK() OVER(order by score desc) as dense_r

from scores_tb;

a6e3bf425fc142049b9add0898523285.png


并列排名,排名有间隔

select xuehao,score,

RANK() over(order by score desc) as r

from scores_tb;


ad473b25c8f442e5918f2ae8e8d1b8f5.png

目录
相关文章
|
9天前
|
SQL NoSQL Java
Java使用sql查询mongodb
通过使用 MongoDB Connector for BI 和 JDBC,开发者可以在 Java 中使用 SQL 语法查询 MongoDB 数据库。这种方法对于熟悉 SQL 的团队非常有帮助,能够快速实现对 MongoDB 数据的操作。同时,也需要注意到这种方法的性能和功能限制,根据具体应用场景进行选择和优化。
33 9
|
29天前
|
SQL 存储 人工智能
Vanna:开源 AI 检索生成框架,自动生成精确的 SQL 查询
Vanna 是一个开源的 Python RAG(Retrieval-Augmented Generation)框架,能够基于大型语言模型(LLMs)为数据库生成精确的 SQL 查询。Vanna 支持多种 LLMs、向量数据库和 SQL 数据库,提供高准确性查询,同时确保数据库内容安全私密,不外泄。
103 7
Vanna:开源 AI 检索生成框架,自动生成精确的 SQL 查询
|
2月前
|
SQL Java
使用java在未知表字段情况下通过sql查询信息
使用java在未知表字段情况下通过sql查询信息
39 8
|
2月前
|
SQL 安全 PHP
PHP开发中防止SQL注入的方法,包括使用参数化查询、对用户输入进行过滤和验证、使用安全的框架和库等,旨在帮助开发者有效应对SQL注入这一常见安全威胁,保障应用安全
本文深入探讨了PHP开发中防止SQL注入的方法,包括使用参数化查询、对用户输入进行过滤和验证、使用安全的框架和库等,旨在帮助开发者有效应对SQL注入这一常见安全威胁,保障应用安全。
63 4
|
2月前
|
SQL 监控 关系型数据库
SQL语句当前及历史信息查询-performance schema的使用
本文介绍了如何使用MySQL的Performance Schema来获取SQL语句的当前和历史执行信息。Performance Schema默认在MySQL 8.0中启用,可以通过查询相关表来获取详细的SQL执行信息,包括当前执行的SQL、历史执行记录和统计汇总信息,从而快速定位和解决性能瓶颈。
|
2月前
|
SQL 存储 缓存
如何优化SQL查询性能?
【10月更文挑战第28天】如何优化SQL查询性能?
168 10
|
2月前
|
SQL 关系型数据库 MySQL
|
3月前
|
SQL 数据库 开发者
功能发布-自定义SQL查询
本期主要为大家介绍ClkLog九月上线的新功能-自定义SQL查询。
|
3月前
|
SQL 移动开发 Oracle
SQL语句实现查询连续六天数据的方法与技巧
在数据库查询中,有时需要筛选出符合特定时间连续性条件的数据记录
|
3月前
|
SQL Java 数据库连接
如何使用`DriverManager.getConnection()`连接数据库,并利用`PreparedStatement`执行参数化查询,有效防止SQL注入。
【10月更文挑战第6天】在代码与逻辑交织的世界中,我从一名数据库新手出发,通过不断探索与实践,最终成为熟练掌握JDBC的开发者。这段旅程充满挑战与惊喜,从建立数据库连接到执行SQL语句,再到理解事务管理和批处理等高级功能,每一步都让我对JDBC有了更深的认识。示例代码展示了如何使用`DriverManager.getConnection()`连接数据库,并利用`PreparedStatement`执行参数化查询,有效防止SQL注入。
147 5