共计 4310 个字符,预计需要花费 11 分钟才能阅读完成。
本篇文章是 SQL 必知必会 的读书笔记,记录如下
- 原文链接: SQL 必知必会
检索数据
-- 检索单列
select name from student;
-- 检索多列
select name, age, class from student;
-- 检索所有列
select * from student;
-- 对某列去重
select distinct class from student;
-- 检索列 - 选择区间
-- offset 基数为 0,所以 `offset 1` 代表从第 2 行开始
select * from student limit 1, 10;
select * from student limit 10 offset 1;
排序
默认排序是 ASC
,所以一般升序的时候不需指定,降序的关键字是 DESC
。
使用 B-Tree
索引可以提高排序性能,但只限最左匹配。
-- 根据学号降序排列
select * from student order by number desc;
-- 添加索引 (score, name) 可以提高排序性能
-- 但是索引 (name, score) 对性能毫无帮助,此谓最左匹配,可以根据 B+Tree 进行理解
select * from student order by score desc, name;
数据过滤
数据筛选,或者数据过滤在 sql 中使用频率最高
-- 找到学号为 1 的学生
select * from student where number = 1;
-- 找到学号为在 [1, 10] 的学生(闭区间)
select * from student where number between 1 and 10;
-- 找到未设置电子邮箱的学生
-- 注意不能使用 =
select * from student where email is null;
-- 找到一班中大于 23 岁的学生
select * from student where class_id = 1 and age > 23;
-- 找到一班或者大于 23 岁的学生
select * from student where class_id = 1 or age > 22;
-- 找到一班与二班的学生
select * from student where class_id in (1, 2);
-- 找到不是一班二班的学生
select * from student where class_id not in (1, 2);
计算字段
-
CONCAT
select concat(name, '(', age, ')') as nameWithAge from student; select concat('hello', 'world') as helloworld;
-
Math
select age - 18 as relativeAge from student; select 3 * 4 as n;
更多函数可以查看 API 手册,同时也可以自定义函数(User Define Function)。
可以直接使用 select
调用函数
select now();
select concat('hello', 'world');
数据聚合 (aggregation)
聚合函数,一些对数据进行汇总的函数,常见有 COUNT
,MIN
,MAX
,AVG
,SUM
五种。
-- 统计 1 班人数
select count(*) from student where class_id = 1;
数据分组
使用 group by
进行数据分组,可以使用聚合函数对分组数据进行汇总,使用 having
对分组数据进行筛选。
-- 按照班级进行分组并统计各班人数
select class_id, count(*) from student group by class_id;
-- 列出大于三个学生的班级
select class_id, count(*) as cnt from student group by class_id having cnt > 3;
子查询
-- 列出软件工程班级中的学生
select * from student where class_id in (select id from class where name = '软件工程');
联接
虽然两个表拥有公共字段便可以创建联接,但是使用外键可以更好地保证数据完整性。比如当对一个学生插入一条不存在的班级的时候,便会插入失败。
一般来说,联接比子查询拥有更好的性能。
-- 列出软件工程班级中的学生
select * from student, class
where student.class_id = class.id and class.name = '软件工程';
-
内联接
内联接又叫等值联接。
-- 列出软件工程班级中的学生 select * from student inner join class on student.class_id = class.id where class.name = '软件工程';
-
自联接
自连接就是相同的表进行联接
-- 列出与张三同一班级的学生 select * from student s1 inner join student s2 on s1.class_id = s2.class_id where s1.name = '张三';
-
外联接
外联接分为
left join
与right join
,left join
指左侧永不会为 null,right join
指右侧永不会为 null。-- 列出每个学生的班级,若没有班级则为 null select name, class.name from student left join class on student.class_id = class.id;
插入数据
使用 insert into
向表中插入数据,也可以插入多行。
插入时可以不指定列名,不过严重依赖表中列的顺序关系,推荐指定列名插入数据,并且可以插入部分列。
-- 插入一条数据
insert into student values(8, '陆小凤', 24, 1, 3);
insert into student(name, age, sex, class_id) values(9, '花无缺', 25, 1, 3);
修改数据
在修改重要数据时,务必先 select 确认是否需要操作数据,然后 begin
方便及时 rollback
-
更新
-- 修改张三的班级 update student set class_id = 2 where name = '张三';
-
删除
-- 删除张三的数据 delete from student where name = '张三'; -- 删除表中所有数据 delete from student; -- 更快地删除表中所有数据 truncate table student;
创建表与更新表
-- 创建学生表,注意添加必要的注释
create table student (id int(11) not null auto_increment comment '学生 id',
name varchar(50) not null comment '学生姓名',
age tinyint unsigned default 20 comment '学生年龄',
sex enum('male', 'famale') comment '性别',
score tinyint comment '入学成绩',
class_id int(11) comment '班级',
createTime timestamp default current_timestamp comment '创建时间',
primary key (id),
foreign key (class_id) references class (id)
) comment '学生表';
-- 根据旧表创建新表
create table student_copy as select * from student;
-- 删除 age 列
alter table student drop column age;
-- 添加 age 列
alter table student add column age smallint;
-- 删除学生表
drop table student;
视图
视图是一种虚拟的表,便于更好地在多个表中检索数据,视图也可以作写操作,不过最好作为只读。在需要多个表联接的时候可以使用视图。
create view v_student_with_classname as
select student.name name, class.name class_name
from student left join class
where student.class_id = class.id;
select * from v_student_with_classname;
约束及索引
-
primiry key
任意两行绝对没有相同的主键,且任一行不会有两个主键且主键绝不为空。使用主键可以加快索引。
alter table student add constraint primary key (id);
-
foreign key
外键可以保证数据的完整性。有以下两种情况。
- 插入张三丰 5 班到 student 表中会失败,因为 5 班在 class 表中不存在。
-
class 表删除 3 班会失败,因为陆小凤和楚留香还在 3 班。
alter table student add constraint foreign key (class_id) references class (id);
-
unique key
唯一索引保证该列值是唯一的,但可以允许有 null。
alter table student add constraint unique key (name);
-
check
检查约束可以使列满足特定的条件,如果学生表中所有的人的年龄都应该大于 0。
不过很可惜 mysql 不支持,可以使用 触发器 代替
alter table student add constraint check (age > 0);
-
index
索引可以更快地检索数据,但是降低了更新操作的性能。
create index index_on_student_name on student (name); alter table student add constraint key(name);
触发器
可以在插入,更新,删除行的时候触发事件。
场景:
- 数据约束,比如学生的年龄必须大于 0
- hook,提供数据库级别的 hook
-- 创建触发器
-- 比如 mysql 中没有 check 约束,可以使用创建触发器,当插入数据小于 0 时,置为 0。create trigger reset_age before insert on student for each row
begin
if NEW.age < 0 then
set NEW.age = 0;
end if;
end;
-- 打印触发器列表
show triggers;
存储过程
-- 创建存储过程
create procedure create_student(name varchar(50))
begin
insert into students(name) values (name);
end;
-- 调用存储过程
call create_student('shanyue');
关注我
欢迎关注公众号 山月行,我会定期分享一些前后端以及运维的文章,并且会有技术与生活上的每日回顾与总结,欢迎关注交流