【Java Web】007 - MySQL(DQL & 多表设计)

目录

前言

一、单表设计 -- DQL(重点掌握)

1、DQL介绍

2、DQL语法

3、基本查询 (select xxx from xxx)

4、条件查询(where xxx)

5、聚合函数

6、分组查询(group by xxx [having 过滤条件])

7、排序查询(order by xxx)

8、分页查询(limit)

9、案例

①、案例一:根据需求完成员工管理的条件分页查询

②、案例二:根据需求,完成员工信息的统计

10、小结

二、多表设计 -- DQL(重点掌握)

1、介绍

2、一对多

①、表设计

②、外键约束

③、逻辑外键

3、一对一

4、多对多

5、综合案例

①、案例需求:

②、页面原型

③、设计表结构

④、小结


前言

在上一章的内容中,我们介绍了:

- 使用DDL语句来操作数据库以及表结构(数据库设计)

- 使用DML语句来完成数据库中数据的增、删、改操作(数据库操作)

在这一章中主要会介绍数据库操作方面的内容:查询(DQL语句)

- DQL语句-单表操作

- DQL语句-多表操作

一、单表设计 -- DQL(重点掌握)

1、DQL介绍

查询操作是所有SQL语句当中最为常见,也是最为重要的操作。在一个正常的业务系统中,查询操作的使用频次是要远高于增删改操作的。当我们打开某个网站或APP所看到的展示信息,都是通过从数据库中查询得到的,而在这个查询过程中,还会涉及到条件、排序、分页等操作。

2、DQL语法

准备一些测试数据(mysql_02.sql)用于查询操作:

create database db02; -- 创建数据库

use db02; -- 切换数据库

-- 员工管理(带约束)

create table tb_emp (

    id int unsigned primary key auto_increment comment 'ID',

    username varchar(20) not null unique comment '用户名',

    password varchar(32) default '123456' comment '密码',

    name varchar(10) not null comment '姓名',

    gender tinyint unsigned not null comment '性别, 说明: 1 男, 2 女',

    image varchar(300) comment '图像',

    job tinyint unsigned comment '职位, 说明: 1 班主任,2 讲师, 3 学工主管, 4 教研主管',

    entrydate date comment '入职时间',

    create_time datetime not null comment '创建时间',

    update_time datetime not null comment '修改时间'

) comment '员工表';

-- 准备测试数据

INSERT INTO tb_emp (id, username, password, name, gender, image, job, entrydate, create_time, update_time) VALUES

    (1, 'jinyong', '123456', '金庸', 1, '1.jpg', 4, '2000-01-01', '2022-10-27 16:35:33', '2022-10-27 16:35:35'),

    (2, 'zhangwuji', '123456', '张无忌', 1, '2.jpg', 2, '2015-01-01', '2022-10-27 16:35:33', '2022-10-27 16:35:37'),

    (3, 'yangxiao', '123456', '杨逍', 1, '3.jpg', 2, '2008-05-01', '2022-10-27 16:35:33', '2022-10-27 16:35:39'),

    (4, 'weiyixiao', '123456', '韦一笑', 1, '4.jpg', 2, '2007-01-01', '2022-10-27 16:35:33', '2022-10-27 16:35:41'),

    (5, 'changyuchun', '123456', '常遇春', 1, '5.jpg', 2, '2012-12-05', '2022-10-27 16:35:33', '2022-10-27 16:35:43'),

    (6, 'xiaozhao', '123456', '小昭', 2, '6.jpg', 3, '2013-09-05', '2022-10-27 16:35:33', '2022-10-27 16:35:45'),

    (7, 'jixiaofu', '123456', '纪晓芙', 2, '7.jpg', 1, '2005-08-01', '2022-10-27 16:35:33', '2022-10-27 16:35:47'),

    (8, 'zhouzhiruo', '123456', '周芷若', 2, '8.jpg', 1, '2014-11-09', '2022-10-27 16:35:33', '2022-10-27 16:35:49'),

    (9, 'dingminjun', '123456', '丁敏君', 2, '9.jpg', 1, '2011-03-11', '2022-10-27 16:35:33', '2022-10-27 16:35:51'),

    (10, 'zhaomin', '123456', '赵敏', 2, '10.jpg', 1, '2013-09-05', '2022-10-27 16:35:33', '2022-10-27 16:35:53'),

    (11, 'luzhangke', '123456', '鹿杖客', 1, '11.jpg', 2, '2007-02-01', '2022-10-27 16:35:33', '2022-10-27 16:35:55'),

    (12, 'hebiweng', '123456', '鹤笔翁', 1, '12.jpg', 2, '2008-08-18', '2022-10-27 16:35:33', '2022-10-27 16:35:57'),

    (13, 'fangdongbai', '123456', '方东白', 1, '13.jpg', 1, '2012-11-01', '2022-10-27 16:35:33', '2022-10-27 16:35:59'),

    (14, 'zhangsanfeng', '123456', '张三丰', 1, '14.jpg', 2, '2002-08-01', '2022-10-27 16:35:33', '2022-10-27 16:36:01'),

    (15, 'yulianzhou', '123456', '俞莲舟', 1, '15.jpg', 2, '2011-05-01', '2022-10-27 16:35:33', '2022-10-27 16:36:03'),

    (16, 'songyuanqiao', '123456', '宋远桥', 1, '16.jpg', 2, '2010-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:05'),

    (17, 'chenyouliang', '12345678', '陈友谅', 1, '17.jpg', null, '2015-03-21', '2022-10-27 16:35:33', '2022-10-27 16:36:07'),

    (18, 'zhang1', '123456', '张一', 1, '2.jpg', 2, '2015-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:09'),

    (19, 'zhang2', '123456', '张二', 1, '2.jpg', 2, '2012-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:11'),

    (20, 'zhang3', '123456', '张三', 1, '2.jpg', 2, '2018-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:13'),

    (21, 'zhang4', '123456', '张四', 1, '2.jpg', 2, '2015-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:15'),

    (22, 'zhang5', '123456', '张五', 1, '2.jpg', 2, '2016-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:17'),

    (23, 'zhang6', '123456', '张六', 1, '2.jpg', 2, '2012-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:19'),

    (24, 'zhang7', '123456', '张七', 1, '2.jpg', 2, '2006-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:21'),

    (25, 'zhang8', '123456', '张八', 1, '2.jpg', 2, '2002-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:23'),

    (26, 'zhang9', '123456', '张九', 1, '2.jpg', 2, '2011-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:25'),

    (27, 'zhang10', '123456', '张十', 1, '2.jpg', 2, '2004-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:27'),

    (28, 'zhang11', '123456', '张十一', 1, '2.jpg', 2, '2007-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:29'),

    (29, 'zhang12', '123456', '张十二', 1, '2.jpg', 2, '2020-01-01', '2022-10-27 16:35:33', '2022-10-27 16:36:31');

3、基本查询 (select xxx from xxx)

【提示】:`*`号代表查询所有字段,在实际开发中尽量少用(不直观、影响效率)

示例:

案例1:查询指定字段 name,entrydate并返回

select name,entrydate from tb_emp;

效果展示:

案例2:查询返回所有字段

select * from tb_emp;

效果展示:

案例3:查询所有员工的 name,entrydate,并起别名(姓名、入职日期)

-- 方式1:

select name AS 姓名, entrydate AS 入职日期 from tb_emp;

-- 方式2: 别名中有特殊字符时,使用''或""包含

select name AS '姓 名', entrydate AS '入职日期' from tb_emp;

-- 方式3:

select name AS "姓名", entrydate AS "入职日期" from tb_emp;

效果展示:

案例4:查询已有的员工关联了哪几种职位(不要重复)

select distinct job from tb_emp;

效果展示:

4、条件查询(where xxx)

示例:

案例1:查询 姓名 为 杨逍 的员工

案例2:查询 id小于等于5 的员工信息

案例3:查询 没有分配职位 的员工信息

注意】:查询为NULL的数据时,不能使用 ‘= null’,而要写成‘is null

案例4:查询 有职位 的员工信息

案例5:查询 密码不等于 '123456' 的员工信息

案例6:查询 入职日期 在 '2000-01-01' (包含) 到 '2010-01-01'(包含) 之间的员工信息

案例7:查询 入职时间 在 '2000-01-01' (包含) 到 '2010-01-01'(包含) 之间 且 性别为女 的员工信息

案例8:查询 职位是 2 (讲师), 3 (学工主管), 4 (教研主管) 的员工信息

案例9:查询 姓名 为两个字的员工信息

案例10:查询 姓 '张' 的员工信息

5、聚合函数

分组查询一般都会伴随着聚合函数,之前我们做的查询都是横向查询,就是根据条件一行一行的进行判断,而使用聚合函数查询就是纵向查询,它是对一列的值进行计算,然后返回一个结果值。(将一列数据作为一个整体,进行纵向计算)

注意 : 聚合函数忽略空值,对NULL值不作为统计

示例:

案例1:统计该企业员工数量

案例2:统计该企业最早入职的员工

案例3:统计该企业最迟入职的员工

案例4:统计该企业员工 ID 的平均值

案例5:统计该企业员工的 ID 之和

6、分组查询(group by xxx [having 过滤条件])

注意】:分组查询后,select之后能够返回的字段包括两类,一类是分组字段,一类是聚合函数

示例:

案例1:根据性别分组 , 统计男性和女性员工的数量

效果展示:

案例2:查询入职时间在 '2015-01-01' (包含) 以前的员工 , 并对结果根据职位分组 , 获取员工数量大于等于2的职位

效果展示:

面试题:where having 的区别

7、排序查询(order by xxx)

排序方式只有两种:升序 & 降序

示例:

案例1:根据入职时间, 对员工进行升序排序

案例2:根据入职时间,对员工进行降序排序

案例3:根据入职时间对公司的员工进行升序排序,入职时间相同,再按照更新时间进行降序排序

注意事项:如果是多字段排序,当第一个字段值相同时,才会根据第二个字段进行排序

8、分页查询(limit)

示例:

案例1:从起始索引0开始查询员工数据, 每页展示5条记录

案例2:查询 第1页 员工数据, 每页展示5条记录

案例3:查询 第2页 员工数据, 每页展示5条记录

案例4:查询 第3页 员工数据, 每页展示5条记录

起始索引计算公式:起始索引 = (查询页码 - 1* 每页显示记录数

 

9、案例

①、案例一:根据需求完成员工管理的条件分页查询

示例:

效果展示:

②、案例二:根据需求,完成员工信息的统计

 

相应的图形报表示例可参考 ECharts(报表组件库)

 

示例:

案例2-1 员工性别统计:

效果展示:

案例2-2 员工职位统计:

效果展示:

 

10、小结

 

注意】:如果写了别名,查询结果将会以别名作为字段名进行显示

二、多表设计 -- DQL(重点掌握)

1、介绍

2、一对多

①、表设计

表结构设计:(部门表)

部门表 - SQL语句:

部门表创建好之后,我们还需要再修改下员工表。为什么要修改员工表呢?是因为我们之前设计员工表(单表)的时候,并没有考虑员工的归属部门

员工表:添加归属部门字段

②、外键约束

问题:现在员工表中有五个员工都归属于1号部门(学工部),当删除了1号部门后,数据变为:

即,1号部门被删除了,但是依然还有5个员工是属于1号部门的。 此时:就出现数据的不完整、不一致

分析:目前上述的两张表(员工表、部门表),在数据库层面,并未建立关联,所以是无法保证数据的一致性和完整性的

解决方法:我们就可以通过数据库中的 外键约束 来解决

外键约束:让两张表的数据建立连接,保证数据的一致性和完整性。 

对应的关键字:foreign key

 

图形化界面方式:

当我们添加了外键之后,再删除ID为1的部门,就会发现,此时数据库报错了,不允许删除,此时仅允许删除没有被关联的id数据:

③、逻辑外键

逻辑外键:又叫事物外键,不使用foreignkey,使用语法(代码)上产生逻辑关联而产生的外键;

两张表使用物理外键进行关联:

两张表使用逻辑外键进行关联:

3、一对一

一对一关系表在实际开发中应用起来比较简单,通常是用来做单表的拆分,也就是将一张大表拆分成两张小表,将大表中的一些基础字段放在一张表当中,将其他的字段放在另外一张表当中,以此来提高数据的操作效率

示例:

SQL脚本

-- 用户基本信息表

create table tb_user(

    id int unsigned  primary key auto_increment comment 'ID',

    name varchar(10) not null comment '姓名',

    gender tinyint unsigned not null comment '性别, 1 男  2 女',

    phone char(11) comment '手机号',

    degree varchar(10) comment '学历'

) comment '用户基本信息表';

-- 测试数据

insert into tb_user values (1,'白眉鹰王',1,'18812340001','初中'),

                        (2,'青翼蝠王',1,'18812340002','大专'),

                        (3,'金毛狮王',1,'18812340003','初中'),

                        (4,'紫衫龙王',2,'18812340004','硕士');

-- 用户身份信息表

create table tb_user_card(

    id int unsigned  primary key auto_increment comment 'ID',

    nationality varchar(10) not null comment '民族',

    birthday date not null comment '生日',

    idcard char(18) not null comment '身份证号',

    issued varchar(20) not null comment '签发机关',

    expire_begin date not null comment '有效期限-开始',

    expire_end date comment '有效期限-结束',

    user_id int unsigned not null unique comment '用户ID',

    constraint fk_user_id foreign key (user_id) references tb_user(id)

) comment '用户身份信息表';

-- 测试数据

insert into tb_user_card values (1,'汉','1960-11-06','100000100000100001','朝阳区公安局','2000-06-10',null,1),

        (2,'汉','1971-11-06','100000100000100002','静安区公安局','2005-06-10','2025-06-10',2),

        (3,'汉','1963-11-06','100000100000100003','昌平区公安局','2006-06-10',null,3),

        (4,'回','1980-11-06','100000100000100004','海淀区公安局','2008-06-10','2028-06-10',4);

4、多对多

多对多的关系在开发中属于也比较常见的。比如:学生和老师的关系,一个学生可以有多个授课老师,一个授课老师也可以有多个学生。在比如:学生和课程的关系,一个学生可以选修多门课程,一个课程也可以供多个学生选修

示例:

SQL脚本:

-- 学生表

create table tb_student(

    id int auto_increment primary key comment '主键ID',

    name varchar(10) comment '姓名',

    no varchar(10) comment '学号'

) comment '学生表';

-- 学生表测试数据

insert into tb_student(name, no) values ('黛绮丝', '2000100101'),('谢逊', '2000100102'),('殷天正', '2000100103'),('韦一笑', '2000100104');

-- 课程表

create table tb_course(

   id int auto_increment primary key comment '主键ID',

   name varchar(10) comment '课程名称'

) comment '课程表';

-- 课程表测试数据

insert into tb_course (name) values ('Java'), ('PHP'), ('MySQL') , ('Hadoop');

-- 学生课程表(中间表)

create table tb_student_course(

   id int auto_increment comment '主键' primary key,

   student_id int not null comment '学生ID',

   course_id  int not null comment '课程ID',

   constraint fk_courseid foreign key (course_id) references tb_course (id),

   constraint fk_studentid foreign key (student_id) references tb_student (id)

)comment '学生课程中间表';

-- 学生课程表测试数据

insert into tb_student_course(student_id, course_id) values (1,1),(1,2),(1,3),(2,2),(2,3),(3,4);

Diagrams -> Show Visualization:

5、综合案例

①、案例需求:

②、页面原型

依次类推进行分析……

表结构关系图:

③、设计表结构

分类表:

菜品表:

套餐表:

套餐菜品关联表:

④、小结

猜你喜欢

转载自blog.csdn.net/qq_41071754/article/details/129932512
今日推荐