三、MySQL从入门到精通---使用DQL查询数据(核心重点)

在开始学习之前,要有一个MySQL的诉苦局进行测试,一下就是数据库数据,大家先要把数据库的内容编写出来,才能进行接下来的学习。

下面展示一些 测试应用代码

CREATE DATABASE IF NOT EXISTS `school`;
-- 创建一个school数据库
USE `school`;-- 创建学生表
DROP TABLE IF EXISTS `student`;
-- ----------------------------
-- Table structure for student
-- ----------------------------
DROP TABLE IF EXISTS `student`;
CREATE TABLE `student`  (
  `studentno` INT(4) NOT NULL COMMENT '学号',
  `loginpwd` VARCHAR(20) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `studentname` VARCHAR(20) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '学生姓名',
  `sex` TINYINT(1) NULL DEFAULT NULL COMMENT '性别,0或1',
  `gradeid` INT(11) NULL DEFAULT NULL COMMENT '年级编号',
  `phone` VARCHAR(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '联系电话,允许为空',
  `address` VARCHAR(255) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '地址,允许为空',
  `borndate` DATETIME(0) NULL DEFAULT NULL COMMENT '出生时间',
  `email` VARCHAR(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '邮箱账号允许为空',
  `identitycard` VARCHAR(18) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '身份证号',
  PRIMARY KEY (`studentno`) USING BTREE,
  UNIQUE INDEX `identitycard`(`identitycard`) USING BTREE,
  INDEX `email`(`email`) USING BTREE
) ENGINE = MYISAM AUTO_INCREMENT = 1 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = DYNAMIC;

`student``student`
-- ----------------------------
-- Table structure for subject
-- ----------------------------
DROP TABLE IF EXISTS `subject`;
CREATE TABLE `subject`  (
  `subjectno` INT(11) NOT NULL AUTO_INCREMENT COMMENT '课程编号',
  `subjectname` VARCHAR(50) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '课程名称',
  `classhour` INT(4) NULL DEFAULT NULL COMMENT '学时',
  `gradeid` INT(4) NULL DEFAULT NULL COMMENT '年级编号',
  PRIMARY KEY (`subjectno`) USING BTREE
) ENGINE = INNODB AUTO_INCREMENT = 19 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = DYNAMIC;

-- ----------------------------
-- Records of subject
-- ----------------------------
INSERT INTO `subject` VALUES (1, '高等数学-1', 110, 1);
INSERT INTO `subject` VALUES (2, '高等数学-2', 110, 2);
INSERT INTO `subject` VALUES (3, '高等数学-3', 100, 3);
INSERT INTO `subject` VALUES (4, '高等数学-4', 130, 4);
INSERT INTO `subject` VALUES (5, 'C语言-1', 110, 1);
INSERT INTO `subject` VALUES (6, 'C语言-2', 110, 2);
INSERT INTO `subject` VALUES (7, 'C语言-3', 100, 3);
INSERT INTO `subject` VALUES (8, 'C语言-4', 130, 4);
INSERT INTO `subject` VALUES (9, 'Java程序设计-1', 110, 1);
INSERT INTO `subject` VALUES (10, 'Java程序设计-2', 110, 2);
INSERT INTO `subject` VALUES (11, 'Java程序设计-3', 100, 3);
INSERT INTO `subject` VALUES (12, 'Java程序设计-4', 130, 4);
INSERT INTO `subject` VALUES (13, '数据库结构-1', 110, 1);
INSERT INTO `subject` VALUES (14, '数据库结构-2', 110, 2);
INSERT INTO `subject` VALUES (15, '数据库结构-3', 100, 3);
INSERT INTO `subject` VALUES (16, '数据库结构-4', 130, 4);
INSERT INTO `subject` VALUES (17, 'C#基础', 130, 1);

SET FOREIGN_KEY_CHECKS = 1;


-- ----------------------------
-- Table structure for result
-- ----------------------------
DROP TABLE IF EXISTS `result`;
CREATE TABLE `result`  (
  `studentno` INT(4) NOT NULL COMMENT '学号',
  `subjectno` INT(4) NOT NULL COMMENT '课程编号',
  `examdate` DATETIME(0) NOT NULL COMMENT '考试日期',
  `studentresult` INT(4) NOT NULL COMMENT '考试成绩',
  INDEX `subjectno`(`subjectno`) USING BTREE
) ENGINE = INNODB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = DYNAMIC;

-- ----------------------------
-- Records of result
-- ----------------------------
INSERT INTO `result` VALUES (1000, 1, '2013-11-11 16:00:00', 85);
INSERT INTO `result` VALUES (1000, 2, '2013-11-12 16:00:00', 70);
INSERT INTO `result` VALUES (1000, 3, '2013-11-11 09:00:00', 68);
INSERT INTO `result` VALUES (1000, 4, '2013-11-13 16:00:00', 98);
INSERT INTO `result` VALUES (1000, 5, '2013-11-14 16:00:00', 58);
INSERT INTO `result` VALUES (1001, 1, '2013-11-11 16:00:00', 75);
INSERT INTO `result` VALUES (1001, 2, '2013-11-12 16:00:00', 70);
INSERT INTO `result` VALUES (1001, 3, '2013-11-11 09:00:00', 68);
INSERT INTO `result` VALUES (1001, 4, '2013-11-13 16:00:00', 98);
INSERT INTO `result` VALUES (1001, 5, '2013-11-14 16:00:00', 58);

-- ----------------------------
-- Table structure for grade
-- ----------------------------
DROP TABLE IF EXISTS `grade`;
CREATE TABLE `grade`  (
  `gradeid` INT(11) NOT NULL AUTO_INCREMENT COMMENT '年级编号',
  `gradename` VARCHAR(50) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '年级名称',
  PRIMARY KEY (`gradeid`) USING BTREE
) ENGINE = INNODB AUTO_INCREMENT = 6 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = DYNAMIC;

-- ----------------------------
-- Records of grade
-- ----------------------------
INSERT INTO `grade` VALUES (1, '大一');
INSERT INTO `grade` VALUES (2, '大二');
INSERT INTO `grade` VALUES (3, '大三');
INSERT INTO `grade` VALUES (4, '大四');
INSERT INTO `grade` VALUES (5, '预科班');

DQL语言

DQL( Data Query Language 数据查询语言 )

  • 查询数据库数据 , 如SELECT语句
  • 简单的单表查询或多表的复杂查询和嵌套查询
  • 是数据库语言中最核心,最重要的语句
  • 使用频率最高的语句

SELECT语法

注意 : [ ] 括号代表可选的 , { }括号代表必选得

SELECT [ALL | DISTINCT]
{* | table.* | [table.field1[as alias1][,table.field2[as alias2]][,…]]}
FROM table_name [as table_alias]
[left | right | inner join table_name2] – 联合查询
[WHERE …] – 指定结果需满足的条件
[GROUP BY …] – 指定结果按照哪几个字段来分组
[HAVING] – 过滤分组的记录必须满足的次要条件
[ORDER BY …] – 指定查询记录按一个或多个条件排序
[LIMIT {[offset,]row_count | row_countOFFSET offset}];
– 指定查询的记录从哪条至哪条

指定查询字段

下面展示一些 SELECT基本用法

-- 查询所有学生信息
SELECT * FROM student

-- 查询指定列(学号,姓名)
SELECT studentno,studentname FROM `student`

AS 子句作为别名

作用:

  • 可给数据列取一个新别名
  • 可给表取一个新别名
  • 可把经计算或总结的结果用另一个新名称来代替

下面展示一些 AS 子句作为别名


-- 给数据列取别名,可以区分不同表格的同名列
SELECT studentno AS 学号, studentname AS 姓名 FROM student;

-- 使用as也可以给表取名
SELECT studentno AS 学号,studentname AS 姓名 FROM student AS s;

-- 使用as为查询结果取一个新的名字
-- CONCAT()函数拼接字符串
SELECT CONCAT ('姓名: ', studentname) AS 新姓名 FROM student;

DISTINCT关键字的使用

作用 : 去掉SELECT查询返回的记录结果中重复的记录 ( 返回所有列的值都相同 ) , 只返回一条

下面展示一些 DISTINCT关键字的使用

-- # 查看那些同学参加了考试(去除重复项)
SELECT * FROM result;-- 查看考试成绩
SELECT studentno FROM result; -- 查看有哪些同学参加了考试
SELECT DISTINCT studentno FROM result; -- 了解:DISTINCT去除重复项,(默认选择全部)

使用表达式的列

数据库中的表达式 : 一般由文本值 , 列值 , NULL , 函数和操作符等组成

应用场景 :

  • SELECT语句返回结果列中使用
  • SELECT语句中的ORDER BY , HAVING等子句中使用
  • DML语句中的 where 条件语句中使用表达式

下面展示一些 内联代码片

// A code block
var foo = 'bar';
// An highlighted block
var foo = 'bar';

where条件语句

作用:用于检索数据表中 符合条件 的记录
搜索条件可由一个或多个逻辑表达式组成 , 结果一般为真或假.

逻辑操作符

在这里插入图片描述

下面展示一些 逻辑操作符

-- 满足条件的查询(where)
SELECT Studentno,StudentResult FROM result;

-- 查询考试成绩在95-100之间的
SELECT Studentno,StudentResult
FROM result
WHERE StudentResult>=95 AND StudentResult<=100;

-- AND也可以写成 &&
SELECT Studentno,StudentResult
FROM result
WHERE StudentResult>=95 && StudentResult<=100;

-- 模糊查询(对应的词:精确查询)
SELECT Studentno,StudentResult
FROM result
WHERE StudentResult BETWEEN 95 AND 100;

-- 除了1000号同学,要其他同学的成绩
SELECT studentno,studentresult
FROM result
WHERE studentno!=1000;

-- 使用NOT
SELECT studentno,studentresult
FROM result
WHERE NOT studentno=1000;

模糊查询 :比较操作符

在这里插入图片描述
注意:

  • 数值数据类型的记录之间才能进行算术运算 ;
  • 相同数据类型的数据之间才能进行比较 ;

下面展示一些 模糊查询 :比较操作符

-- 模糊查询BETWEEN / AND / LIKE / IN /NULL
-- =============================================
-- LIKE
-- =============================================
-- 查询姓赵的同学的学号及姓名
-- like结合使用的通配符 : % (代表0到任意个字符) _ (一个字符)
SELECT studentno,studentname FROM student
WHERE studentname LIKE '赵%';

-- 查询姓刘的同学,后面只有一个字的
SELECT studentno,studentname FROM student
WHERE studentname LIKE '赵_';

-- 查询姓刘的同学,后面只有两个字的
SELECT studentno,studentname FROM student
WHERE studentname LIKE '赵__';

-- 查询姓名中含有强字的
SELECT studentno,studentname FROM student
WHERE studentname LIKE '%强%';

-- 查询姓名中含有特殊字符的需要使用转义符号 '\'
-- 自定义转义符关键字: ESCAPE ':'

-- ==================================
-- IN
-- ==================================
-- 查询学号为1000,1001,1003的学生姓名
SELECT studentno,studentname FROM student WHERE studentno IN (1000,1001,1003);

-- 查询地址在北京朝阳,安徽芜湖的学生
SELECT studentno,studentname,address FROM student WHERE address IN ('北京朝阳','安徽芜湖');


-- ==================================
-- NULL
-- ==================================

-- 查询出生日期没有填写的同学
-- 不能直接写=NULL , 这是代表错误的 , 用 is null
SELECT studentname FROM student WHERE borndate IS NOT NULL;

SELECT studentname FROM student WHERE borndate IS NULL;

-- 查询没有写家庭住址的同学(空字符串不等于null)
SELECT studentname FROM student WHERE address = ' ' OR address IS NULL;

连接查询

连接查询:如果需要多张数据表的数据进行查询,则通过连接运算符来实现多个查询

  • 内连接 INNER JOIN :查询两个表中的结果集中的交集
  • 外连接 OUTER JOIN :
  • 左外连接 OUTER JOIN:以左表作为基准,右表来一一匹配,匹配不上的返回左表的记录,右表以NULL填充
  • 右外连接 RIGHT JOIN : 以右作为基准,左表来一一匹配,匹配不上的返回右表的记录,左表以NULL填充

JOIN 对比

在这里插入图片描述
在这里插入图片描述
下面展示一些 JOIN 对比

SELECT * FROM result;
SELECT * FROM student;

/*思路:
(1)分析要求,确定查询的列来源于那两个表,
(2)确定选择哪种连接查询
*/
-- 内连接,指导出student和result在studentno相同所对应的数据
SELECT s.studentno ,studentname,subjectno,studentresult FROM student s 
INNER JOIN result r 
ON r.studentno = s.studentno 

-- 右连接,将result中所有的元素都导出来,包括缺考人员
SELECT s.studentno,studentname,subjectno,studentresult FROM student s
RIGHT JOIN result r
ON r.studentno = s.studentno
-- WHERE r.studentresult = 0

-- 左连接,指导出student中的数据
SELECT s.studentno,studentname,subjectno,studentresult FROM student s
LEFT JOIN result r
ON r.studentno = s.studentno
WHERE r.studentresult BETWEEN 85 AND 100


-- 思考题:查询参加考试的同学信息(学号,学生姓名,科目名,分数)
SELECT s.studentno,studentname,subjectname,studentresult FROM student s
INNER JOIN result r
ON s.studentno = r.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno 
WHERE r.studentresult BETWEEN 85 AND 100

自连接

自连接:数据表与自身进行连接
需求:从一个包含栏目ID , 栏目名称和父栏目ID的表中, 查询父栏目名称和其他子栏目名称

下面展示一些 自连接

USE `school`
-- 创建一个表
DROP TABLE IF  EXISTS `category`;
CREATE TABLE `category` (
`categoryid` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主题id',
`pid` INT(10) NOT NULL COMMENT '父id',
`categoryName` VARCHAR(50) NOT NULL COMMENT '主题名字',
PRIMARY KEY (`categoryid`)
) ENGINE=INNODB AUTO_INCREMENT=9 DEFAULT CHARSET=utf8;

-- 插入数据
INSERT INTO `category` (`categoryid`, `pid`, `categoryName`)
VALUES('2','1','信息技术'),
('3','1','软件开发'),
('4','3','数据库'),
('5','1','美术设计'),
('6','3','web开发'),
('7','5','ps技术'),
('8','2','办公信息');

-- 编写SQL语句,将栏目的父子关系呈现出来 (父栏目名称,子栏目名称)
-- 核心思想:把一张表看成两张一模一样的表,然后将这两张表连接查询(自连接)
SELECT a.categoryid AS '父栏目',b.`categoryName` AS '子栏目' FROM `category` AS a,`category` AS b
WHERE a.`categoryid` = b.`pid`

-- 思考题:查询参加了考试的同学信息(学号,学生姓名,科目名,分数)
SELECT s.studentno ,studentname,subjectname,studentresult FROM student s 
INNER JOIN  result r
ON s.studentno = r.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno


-- 查询学员及所属的年级(学号,学生姓名,年级名)

SELECT s.studentno AS 学号, studentname AS 学生姓名,gradename AS 年级名 FROM student s
INNER JOIN grade g
ON s.gradeid = g.gradeid

-- 高等数学-4 的所有考试结果(学号,学生姓名 科目名称 成绩)
SELECT s.studentno ,studentname ,subjectname,studentresult FROM student s
INNER JOIN result r
ON s.studentno = r.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno
WHERE subjectname = '高等数学-4'

排序和分页

============== 排序 ================
语法 : ORDER BY

  • ORDER BY 语句用于根据指定的列对结果集进行排序。
  • ORDER BY 语句默认按照ASC升序对记录进行排序。
  • 如果您希望按照降序对记录进行排序,可以使用 DESC 关键字。、

============== 分页 ================
语法 : SELECT * FROM table LIMIT [offset,] rows | rows OFFSET offset
好处 : (用户体验,网络传输,查询压力)

推导:
第一页 : limit 0,5
第二页 : limit 5,5
第三页 : limit 10,5

第N页 : limit (pageNo-1)*pageSzie,pageSzie
[pageNo:页码,pageSize:单页面显示条数]

下面展示一些 排序和分页

============== 排序 ================
-- 高等数学-4 的所有考试结果(学号,学生姓名 科目名称 成绩)按照成绩降序排序
SELECT s.studentno ,studentname ,subjectname,studentresult FROM student s
INNER JOIN result r
ON s.studentno = r.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno
WHERE subjectname = '高等数学-4' ORDER BY studentresult DESC

-- 每页显示5条数据,数据插入较少就不演示了
SELECT s.studentno,studentname,subjectname,StudentResult
FROM student s
INNER JOIN result r
ON r.studentno = s.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno
ORDER BY StudentResult DESC , studentno
LIMIT 0,5

-- 全部课程成绩前10名并且分数大于80的学生信息(学号,姓名,课程名,分数)
SELECT s.studentno,studentname,subjectname,studentresult
FROM student s
INNER JOIN result r
ON r.studentno = s.studentno
INNER JOIN `subject` sub
ON r.subjectno = sub.subjectno
WHERE studentresult >= 50
ORDER BY studentresult DESC
LIMIT 0,10

子查询

============== 子查询 ================
什么是子查询?

  • 在查询语句中的WHERE条件子句中,又嵌套了另一个查询语句
  • 嵌套查询可由多个子查询组成,求解的方式是由里及外;
  • 子查询返回的结果一般都是集合,故而建议使用IN关键字;

下面展示一些 子查询

-- 查询 数据库结构-1 的所有考试结果(学号,科目编号,成绩),并且成绩降序排列
-- 方法一:使用连接查询
SELECT studentno,r.subjectno,StudentResult FROM result r
INNER JOIN `subject` sub
ON r.`subjectNo`=sub.`subjectno`
WHERE subjectname = '数据库结构-1'
ORDER BY studentresult DESC;

-- 方法二:使用子查询(执行顺序:由里及外)
SELECT studentno,subjectno,StudentResult
FROM result
WHERE subjectno=(
   SELECT subjectno FROM `subject`
   WHERE subjectname = '数据库结构-1'
)
ORDER BY studentresult DESC;

-- 查询课程为 高等数学-2 且分数不小于80分的学生的学号和姓名
-- 方法一:使用连接查询
SELECT s.studentno,studentname
FROM student s
INNER JOIN result r
ON s.`StudentNo` = r.`StudentNo`
INNER JOIN `subject` sub
ON sub.`SubjectNo` = r.`SubjectNo`
WHERE subjectname = '高等数学-4' AND StudentResult>=80

-- 方法二:使用连接查询+子查询
-- 分数不小于80分的学生的学号和姓名
SELECT r.studentno,studentname FROM student s
INNER JOIN result r ON s.`StudentNo`=r.`StudentNo`
WHERE StudentResult>=80

-- 在上面SQL基础上,添加需求:课程为 高等数学-4
SELECT r.studentno,studentname FROM student s
INNER JOIN result r ON s.`StudentNo`=r.`StudentNo`
WHERE StudentResult>=80 AND subjectno=(
   SELECT subjectno FROM `subject`
   WHERE subjectname = '高等数学-2'
)

-- 方法三:使用子查询
-- 分步写简单sql语句,然后将其嵌套起来
SELECT studentno,studentname FROM student WHERE studentno IN(
   SELECT studentno FROM result WHERE studentresult>=80 AND subjectno=(
       SELECT subjectno FROM `subject` WHERE subjectname = '高等数学-4'
  )
)
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

原来如此呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值