【MySQL初级】第5-11章
第5章 表相关
5.1创建表
语法格式:
create table 表名(
字段名1 数据类型,
字段名2 数据类型,
字段名3 数据类型,
......
);
例如:创建学生表
create table t_student(
no int,
name varchar,
gender char(1) default '男'
);
5.2 插入数据
语法格式:
insert into 表名(字段名1, 字段名2, 字段名3,......) values (值1,值2,值3,......);
字段名和值要一一对应。类型要一一对应,数量要一一对应。
字段名也可以省略,如果字段名省略就表示把所有字段名都写上去了,并且顺序和建表时的顺序相同。
5.3 删除表
语法格式:
drop table 表名;
或者
drop table if exists 表名;
判断是否存在这个表,如果存在则删除。避免不存在时的报错。
5.4 MySQL数据类型
数据类型(data_type)是指系统中所允许的数据的类型。数据库中的每个列都应该有适当的数据类型,用于限制或允许该列中存储的数据。例如,列中存储的为数字,则相应的数据类型应该为数值类型。
如果使用错误的数据类型可能会严重影响应用程序的功能和性能,所以在设计表时,应该特别重视数据列所用的数据类型。更改包含数据的列不是一件小事,这样做可能会导致数据丢失。因此,在创建表时必须为每个列设置正确的数据类型和长度。
MySQL 的数据类型可以分为整数类型、浮点数类型、定点数类型、日期和时间类型、字符串类型、二进制类型等。
整数类型
tinyint:1个字节(微小整数)
smallint:2个字节(小整数)
mediumint:3个字节(中等大小的整数)
int(integer):4个字节(普通大小整数)
bigint:8个字节(大整数)
浮点数类型
float:4个字节,单精度(最多5位小数)
double:8个字节,双精度(最多16位小数)
定点数类型
decimal:定点数类型。底层实际上采用字符串的形式存储数字。
语法:decimal(m, d)
例如:decimal(3, 2) 表示3个有效数字,2个小数。(有效数字最多65个,小数位最多30个)
日期和时间类型
year:1个字节,只存储年,格式YYYY
time:3个字节,只存储时间,格式HH:MM:SS / HHMMSS
date:3个字节,只存储年月日,格式:YYYY-MM-DD
datetime:8个字节,存储年月日+时分秒,格式:YYYY-MM-DD HH:MM:SS(从公元1000年~公元9999年)
timestamp:4个字节,存储年月日+时分秒,格式:YYYY-MM-DD HH:MM:SS(从公元1980年~公元2040年)或者格式为 YYYYMMDDHHMMSS(采用这种格式不需要使用单引号,当然你使用单引号也可以)
字符串类型
char
char(m): m长度是0~255个字符。
固定长度字符串,在定义时指定字符串列长。当保存时,在右侧填充空格以达到指定的长度。m表示列的长度,范围是 0~255 个字符。
例如,CHAR(4) 定义了一个固定长度的字符串列,包含的字符个数最大为 4。当插入的字符长度大于4,则报错(除非超过4个长度之后都是空格字符,则空格字符会自动被删除用来保证插入的成功)。
varchar
varchar(m): m长度是0~16383个字符
长度可变的字符串。varchar 的最大实际长度由最长的行的大小和使用的字符集确定,而实际占用的空间为字符串的实际长度加 1。
例如,varchar(50) 定义了一个最大长度为 50 的字符串,如果插入的字符串只有 10 个字符,则实际存储的字符串为 10 个字符和一个字符串结束字符。varchar在值保存和检索时尾部的空格仍保留。
text
text类型:
- tinytext 表示长度为 255字符的 TEXT 列。
- text 表示长度为 65535字符的 TEXT 列。
- mediumtext 表示长度为 16777215字符的 TEXT 列。
- longtext 表示长度为 4294967295 或 4GB 字符的 TEXT 列。
enum
enum类型:
- 语法:<字段名> enum(‘值1’,‘值2’,…)
- 该字段插入值时,只能是指定的枚举值。
set
set类型:
- 语法:<字段名> set(‘值1’,‘值2’,‘值3’,…) 注意:值不可重复。
- 该字段插入值时,只能是指定的值。
二进制类型
BLOB类型:二进制大对象,可以存储图片、声音、视频等文件。
- blob:小的,最大长度65535个字节
- mediumblob:中等的,最大长度16777215个字节
- longblob:大的,最大长度4GB的字节
5.5 增删改表结构DDL
创建一个学生表
create table t_student(
no bigint,
name varchar(255),
age int comment '年龄'
);
查看建表语句
show create table 表名;
修改表名
alter table 表名 rename 新表名;
新增字段
alter table 表名 add 字段名 数据类型;
修改字段名
alter table 表名 change 旧字段名 新字段名 数据类型;
修改字段数据类型
alter table 表名 modify column 字段名 数据类型;
删除字段
alter table 表名 drop 字段名;
5.6 DML语句
当我们对表中的数据进行增删改的时候,称它为DML语句。(数据操纵语言),主要包括:insert、delete、update
insert 增
语法格式:
insert into 表名(字段名1,字段名2,字段名3,...) values(值1,值2,值3,...);
表名后面的小括号当中的字段名如果省略掉,表示自动将所有字段都列出来了,并且字段的顺序和建表时的顺序一致。
一般为了可读性强,建议把字段名写上。
insert into 表名 values(值1,值2,值3,...);
一次可以插入多条记录:
insert into t_stu(no,name,age) values(1,'jack',20),(2,'lucy',30);
delete 删
语法格式:
# 将所有记录全部删除
delete from 表名;
# 删除符合条件的记录
delete from 表名 where 条件;
以上的删除属于DML的方式删除,这种删除的数据是可以通过事务回滚的方式重新恢复的,但是删除的效率较低。(这种删除是支持事务的。)
另外还有一种删除表中数据的方式,但是这种方式不支持事务,不可以回滚,删了之后数据是永远也找不回来了。这种删除叫做:表被截断。
注意:这个语句删除效率非常高,巨大的表,瞬间干掉所有数据。但不可恢复。
truncate table 表名;
update 改
语法格式:
update 表名 set 字段名1=值1, 字段名2=值2, 字段名3=值3 where 条件;
如果没有更新条件的话,所有记录全部更新。
5.7 约束constraint
创建表时,可以给表的字段添加约束,可以保证数据的完整性、有效性。比如大家上网注册用户时常见的:用户名不能为空。对不起,用户名已存在。等提示信息。
约束通常包括:
- 非空约束:not null
- 检查约束:check(mysql8之后才支持,oracle一直支持)
- 唯一性约束:unique
- 主键约束:primary key
- 外键约束:foreign key
非空约束
语法格式:
create table t_stu(
no int,
name varchar(255) not null,
age int
);
name字段不能为空。插入数据时如果没有给name指定值,则报错。
检查约束
create table t_stu(
no int,
name varchar(255),
age int,
check(age > 18)
);
唯一性约束
语法格式:
create table t_stu(
no int,
name varchar(255),
email varchar(255) unique
);
email字段设置为唯一性,唯一性的字段值是可以为NULL的。但不能重复。以上在字段后面添加的约束,叫做列级约束。
当然,添加约束还有另一种方式:表级约束:
create table t_stu(
no int,
name varchar(255),
email varchar(255),
unique(email)
);
使用表级约束可以为多个字段添加联合唯一。
create table t_stu(
no int,
name varchar(255),
email varchar(255),
unique(name,email)
);
创建约束时也可以给约束起名字,将来可以通过约束的名字来删除约束:
create table t_stu(
no int,
name varchar(255),
email varchar(255),
constraint t_stu_name_email_unique unique(name,email)
);
所有的约束都存储在一个系统表当中:table_constraints。这个系统表在这个数据库当中:information_schema
主键约束
- 主键:primary key,简称PK
- 主键约束的字段不能为NULL,并且不能重复。
- 任何一张表都应该有主键,没有主键的表可以视为无效表。
- 主键值是这行记录的身份证号,是唯一标识。在数据库表中即使两条数据一模一样,但由于主键值不同,我们也会认为是两条完全的不同的数据。
- 主键分类:
- 根据字段数量分类:
- 单一主键(1个字段作为主键)==>建议的
- 复合主键(2个或2个以上的字段作为主键)
- 根据业务分类:
- 自然主键(主键和任何业务都无关,只是一个单纯的自然数据)===>建议的
- 业务主键(主键和业务挂钩,例如:银行卡账号作为主键)
- 根据字段数量分类:
- 单一主键(建议使用这种方式)
create table t_student(
id bigint primary key,
sno varchar(255) unique,
sname varchar(255) not null
)
- 复合主键(很少用,了解)
create table t_user(
no int,
name varchar(255),
age int,
primary key(no,name)
);
- 主键自增:既然主键值是一个自然的数字,mysql为主键值提供了一种自增机制,不需要我们程序员维护,mysql自动维护该字段
create table t_vip(
no int primary key auto_increment,
name varchar(255)
);
外键约束
- 有这样一个需求:要求设计表,能够存储学生以及学校信息。
- 第一种方案:一张表
这种方式会导致数据冗余,浪费空间。
2. 第二种方案:两张表:一张存储学生,一张存储学校
t_school 表
t_student 表
如果采用以上两张表存储数据,对于学生表来说,sno这个字段的值是不能随便填的,这个sno是学校编号,必须要求这个字段中的值来自学校表的sno。
为了达到要求,此时就必须要给t_student表的sno字段添加外键约束了。
- 外键约束:foreign key,简称FK。
- 添加了外键约束的字段中的数据必须来自其他字段,不能随便填。
- 假设给a字段添加了外键约束,要求a字段中的数据必须来自b字段,b字段不一定是主键,但至少要有唯一性。
- 外键约束可以给单个字段添加,叫做单一外键。也可以给多个字段联合添加,叫做复合外键。复合外键很少用。
- a表如果引用b表中的数据,可以把b表叫做父表,把a表叫做子表。
- 创建表时,先创建父表,再创建子表。
- 插入数据时,先插入父表,在插入子表。
- 删除数据时,先删除子表,再删除父表。
- 删除表时,先删除子表,再删除父表。
- 如何添加外键:
create table t_school(
sno int primary key,
sname varchar(255)
);
create table t_student(
no int primary key,
name varchar(255),
age int,
sno int,
constraint t_school_sno_fk foreign key(sno) references t_school(sno)
);
- 级联删除
创建子表时,外键可以添加:on delete cascade,这样在删除父表数据时,子表会级联删除。谨慎使用。
create table t_student(
no int primary key,
name varchar(255),
age int,
sno int,
constraint t_school_sno_fk foreign key(sno) references t_school(sno) on delete cascade
);
###删除约束
alert table t_student drop foreign key t_student_sno_fk;
###添加约束
alert table t_student add constraint t_student_sno_fk foreign key(sno) references t_school(sno) on delete cascade;
- 级联更新
create table t_student(
no int primary key,
name varchar(255),
age int,
sno int,
constraint t_school_sno_fk foreign key(sno) references t_school(sno) on update cascade
);
- 级联置空
create table t_student(
no int primary key,
name varchar(255),
age int,
sno int,
constraint t_school_sno_fk foreign key(sno) references t_school(sno) on delete set null
);
第6章 三范式★
6.1 什么是数据库设计三范式
数据库表设计的原则。教你怎么设计数据库表有效,并且节省空间。
6.2 三范式
- 第一范式:任何一张表都应该有主键,每个字段是原子性的不能再分
- 以下表的设计不符合第一范式:无主键,并且联系方式可拆分。
2. 应该这样设计:
- 第二范式:建立在第一范式基础上的,另外要求所有非主键字段完全依赖主键,不能产生部分依赖
- 以下表存储了学生和老师的信息
虽然符合第一范式,但是违背了第二范式,学生姓名、老师姓名都产生了部分依赖。导致数据冗余。
2. 以下这种设计方式就是符合第二范式的:
- 第三范式:建立在第二范式基础上的,非主键字段不能传递依赖于主键字段
- 以下设计方式就是违背第三范式的
以上因为产生了传递依赖,导致班级名称冗余。
2. 以下这种方式就是符合第三范式的:
6.3 一对多怎么设计
口诀:一对多两张表,多的表加外键。
6.4 多对多怎么设计
多对多三张表,关系表添加外键。
6.5 一对一怎么设计
两种方案:
- 第一种:主键共享
- 第二种:外键唯一
6.6 最终的设计
最终以满足客户需求为原则,有的时候会拿空间换速度。
第7章 视图
- 只能将select语句创建为视图。
- 创建视图
create or replace view v_emp as select e.ename,d.dname from emp e join dept d on e.deptno = d.deptno;
- 视图作用
- 如果开发中有一条非常复杂的SQL,而这个SQL在多处使用,会给开发和维护带来成本。使用视图可以降低开发和维护的成本。
- 视图可以隐藏表的字段名。
- 修改视图
alter view v_emp as select e.ename,d.dname,d.deptno from emp e join dept d on e.deptno = d.deptno;
- 删除视图
- drop view if exists v_emp;
- 对视图增删改(DML:insert delete update)可以影响到原表数据。
第8章 事务
8.1 事务概述
- 事务是一个最小的工作单元。在数据库当中,事务表示一件完整的事儿。
- 一个业务的完成可能需要多条DML语句共同配合才能完成,例如转账业务,需要执行两条DML语句,先更新张三账户的余额,再更新李四账户的余额,为了保证转账业务不出现问题,就必须保证要么同时成功,要么同时失败,怎么保证同时成功或者同时失败呢?就需要使用事务机制。
- 也就是说用了事务机制之后,在同一个事务当中,多条DML语句会同时成功,或者同时失败,不会出现一部分成功,一部分失败的现象。
- 事务只针对DML语句有效:因为只有这三个语句是改变表中数据的。
- insert
- delete
- update
8.2 事务四大特性:ACID
- 原子性(Atomicity):是指事务包含的所有操作要么全部成功,要么同时失败。
- 一致性(Consistency):是指事务开始前,和事务完成后,数据应该是一致的。例如张三和李四的钱加起来是5000,中间不管进行过多少次的转账操作(update),总量5000是不会变的。这就是事务的一致性。
- 隔离性(Isolation):隔离性是当多个⽤户并发访问数据库时,⽐如操作同⼀张表时,数据库为每⼀个⽤户开启的事务,不能被其他事务的操作所⼲扰,多个并发事务之间要相互隔离。
- 持久性(Durability):持久性是指⼀个事务⼀旦被提交了,那么对数据库中的数据的改变就是永久性的,即便是在数据库系统遇到故障的情况下也不会丢失提交事务的操作。
8.3 演示MySQL事务
在dos命令窗口中开启MySQL事务:start transaction; 或者:begin;
回滚事务:rollback;
提交事务:commit;
只要执行以上的rollback或者commit,事务都会结束。
MySQL默认情况下采用的事务机制是:自动提交。所谓自动提交就是只要执行一条DML语句则提交一次。
8.4 事务隔离级别
隔离级别从低到高排序:读未提交 < 读提交 < 可重复读 < 串行化
不同隔离级别会存在不同的现象,现象按照严重性从高到低排序:脏读 > 不可重复读 > 幻读
查看与设置隔离级别
mysql默认的隔离级别:可重复读(REPEATABLE READ)。
- 查看当前会话的隔离级别:select @@transaction_isolation;
- 查看全局的隔离级别:select @@gobal.transaction_isolation;
设置事务隔离级别:
- 会话级:set session transaction isolation level read committed;
- 全局级:set global transaction isolation level read committed;
不同现象
脏读
指的是一个事务读取了另一个事务尚未提交的数据,即读取了另一个事务中的脏数据(Dirty Data)。在此情况下,如果另一个事务回滚了或者修改了这些数据,那么读取这些脏数据的事务所处理的数据就是不准确的。
不可重复读
指在一个事务内,多次读取同一个数据行,得到的结果可能是不一样的。这是由于其他事务对数据行做出了修改操作,导致数据的不一致性。
幻读
指在事务执行过程中,前后两次相同的查询条件得到的结果集不一致,可能会变多或变少。
(幻读是insert,delete操作,不可重复读是update操作)
隔离级别
读未提交(READ UNCOMMITTED)
A事务与B事务,A事务可以读取到B事务未提交的数据。这是最低的隔离级别。几乎两个事务之间没有隔离。这种隔离级别是一种理论层面的,在实际的数据库产品中,没有从这个级别起步的。
当事务隔离级别是读未提交时,三种现象都存在:脏读,不可重复读,幻读。
我们可以开启两个DOS命令窗口,模拟两个事务,演示一下这种隔离级别。三种现象中最严重的是脏读,我们只需要演示脏读问题即可,因为存在脏读的话,就一定存在不可重复读和幻读问题。
将全局事务隔离级别设置为:READ UNCOMMITTED
set global transaction isolation level read uncommitted;
开启两个DOS命令窗口来模拟两个事务:A事务与B事务。
A事务 | B事务 |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(4); |
| mysql> select * from a;
| |
通过以上测试,可以看到,A事务读取到了B事务还没有提交的数据。这种现象就是脏读。
读提交(READ COMMITTED)
A事务与B事务,A事务可以读取到B事务提交之后的数据。Oracle数据库默认的就是这种隔离级别。
将数据库的全局事务隔离级别设置为读提交:READ COMMITTED
set global transaction isolation level read committed;
演示:
A事务 | B事务 |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(4); |
| mysql> select * from a;
| |
| | mysql> commit; |
| mysql> select * from a;
| |
通过以上测试看出,A事务只能读取到B事务提交之后的数据。这种隔离级别解决了脏读问题,但肯定是存在不可重复读和幻读问题。因为只要事务B进行了增删改操作之后并提交了,事务A读取到的数据肯定是不同的。即:不可重复读和幻读都存在。
可重复读(REPEATABLE READ)
这个隔离级别是MySQL数据库默认的。
A事务和B事务,A事务开启后,读取了某一条记录,然后B事务对这条记录进行修改并提交,A事务读取到的还是修改前的数据。这种隔离级别称为可重复读。
将数据库全局隔离级别修改为可重复读:
set global transaction isolation level repeatable read;
演示:
A事务 | B事务 |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select empno,ename,sal from emp where empno=7369;
| |
| | mysql> update emp set ename=‘SMITH’,sal=8000 where empno=7369; |
| | mysql> commit; |
| mysql> select empno,ename,sal from emp where empno=7369;
| |
通过以上测试得知:当事务隔离级别设置为可重复读时,避免了不可重复读问题。
那么在MySQL当中,当事务隔离级别设置为可重复读时,能够避免幻读问题吗?测试一下:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(5); |
| | mysql> commit; |
| mysql> select * from a;
| |
通过以上测试得知:当事务隔离级别设置为可重复读时,也避免了幻读问题。是完全避免了幻读问题吗?并不是。 请看以下测试:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(6); |
| | mysql> commit; |
| mysql> select * from a for update;
| |
通过以上测试得知:当事务隔离级别设置为可重复读,MySQL会尽最大努力避免幻读问题,但这种隔离级别无法完全避免幻读问题。
串行化(SERIALIZABLE)
这种隔离级别最高,避免了所有的问题,缺点是效率低,因为这种隔离级别会导致事务排队处理,不支持并发。
设置数据库全局隔离级别为串行化:
set global transaction isolation level serializable;
演示:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| mysql> insert into a values(7); | |
| | mysql> select * from a;
|
| mysql> commit; | |
| | |
通过以上测试得知:当事务隔离级别设置为串行化时,事务只能排队执行,不支持并发。
可重复读的幻读问题
在上面讲解过程中我提到,MySQL默认的隔离级别可重复读,在很大程度上避免了幻读问题(并不能完全解决),那么它是如何解决幻读问题的呢,解决方案包括两种:
- 针对快照读(普通 select 语句),是通过 MVCC 方式解决了幻读,因为可重复读隔离级别下,事务执行过程中看到的数据,一直跟这个事务启动时看到的数据是一致的,即使中途有其他事务插入了一条数据,是查询不出来这条数据的,所以就很好的避免了幻读问题。
- 针对当前读(select … for update 等语句),是通过 next-key lock(记录锁+间隙锁)方式解决了幻读,因为当执行 select … for update 语句的时候,会加上 next-key lock,如果有其他事务在 next-key lock 锁范围内插入了一条记录,那么这个插入语句就会被阻塞,无法成功插入,所以就很好的避免了幻读问题。
快照读是如何解决幻读的
什么是快照读?普通的select语句都是采用的快照读。顾名思义:在整个事务的处理过程中,执行相同的一个select语句时,每次都是读取的快照。(快照指的是固定的某个时刻的数据,就像现实世界中的拍照一样,把那个美好的时刻留下来)。也就是说,当事务隔离级别是可重复读,并且执行的select语句是一个普通的select语句时,都会采用快照读的方式读取数据,底层实现原理是:
- 底层由 MVCC(多版本并发控制)实现,实现的方式是开始事务后,在执行第一个查询语句后,会创建一个 Read View,后续的查询语句利用这个 Read View,通过这个 Read View 就可以在 undo log 版本链找到事务开始时的数据,所以事务过程中每次查询的数据都是一样的,即使中途有其他事务插入了新纪录,是查询不出来这条数据的,所以就很好的避免了幻读问题。
演示:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a; //快照读
| |
| | mysql> insert into a values(5); |
| | mysql> commit; |
| mysql> select * from a; //快照读
| |
当前读是如何解决幻读的
当前读,顾名思义:每一次都读取最新的数据。当前读包括:update、delete、insert、select…for update。这个很好理解,因为增删改的时候都要基于最新的数据进行增删改。
而select…for update原理是:对查询范围内的数据进行加锁,不允许其它事务对这个范围内的数据进行增删改。也就是说这个select语句范围内的数据是不允许并发的,只能排队执行,从而避免幻读问题。
select…for update加的锁叫做:next-key lock。我们可以称其为:间隙锁 + 记录锁。间隙锁用来保证在锁定的范围内不允许insert操作。记录锁用来保证在锁定的范围内不允许delete和update操作。
假如有这样的数据:
SQL语句是这样写的:
select * from a where id between 2 and 4 for update;
那么id在[2-4]区间的所有记录行被锁定,不能插入3是通过间隙锁来搞定的。不能修改或删除2和4是通过记录锁来搞定的。
演示:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; | |
mysql> select * from a where id between 2 and 4 for update; // 当前读 | |
出现幻读的两种情况
在同一个事务处理过程中,如果前后两次都采用快照读,或者都采用当前读,则不会出现幻读问题。如果第一次使用快照读,后面使用了当前读,则会出现幻读问题。
第一种产生幻读的场景
A事务与B事务。在A事务中第一次查询使用快照读,B事务插入数据。然后在A事务中第二次查询使用当前读。则会产生幻读现象。
演示:
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(5); |
| | mysql> commit; |
| mysql> select * from a for update; // 产生了幻读
| |
第二种产生幻读的场景
事务A与事务B,在事务A中第一次查询使用快照读,在事务B中插入一条数据,然后在事务A中更新事务B插入的那条记录,最后在事务A中再次使用快照读。则会发生幻读现象。
事务A | 事务B |
---|---|
mysql> use powernode | |
mysql> use powernode | |
mysql> start transaction; | |
mysql> start transaction; |
| mysql> select * from a;
| |
| | mysql> insert into a values(6); |
| | mysql> commit; |
| mysql> update a set id=100 where id=6; //主要是因为这个SQL语句的执行触发了当前读 | |
| mysql> select * from a; // 产生了幻读
| |
总结可重复读的幻读问题
MySQL的可重复读隔离级别(默认隔离级),根据不同的查询方式,分别提出了避免幻读的方案:
- 针对快照读(普通 select 语句),是通过 MVCC 方式解决了幻读。
- 针对当前读(select … for update 等语句),是通过 next-key lock(记录锁+间隙锁)方式解决了幻读。
我举例了两个发生幻读场景的例子。
- 第一个例子:对于快照读, MVCC 并不能完全避免幻读现象。因为当事务 A 更新了一条事务 B 插入的记录,那么事务 A 前后两次查询的记录条目就不一样了,所以就发生幻读。
- 第二个例子:对于当前读,如果事务开启后,并没有执行当前读,而是先快照读,然后这期间如果其他事务插入了一条记录,那么事务后续使用当前读进行查询的时候,就会发现两次查询的记录条目就不一样了,所以就发生幻读。
所以,MySQL 可重复读隔离级别并没有彻底解决幻读,只是很大程度上避免了幻读现象的发生。
要避免这类特殊场景下发生幻读的现象的话,就是尽量在开启事务之后,马上执行 select … for update 这类当前读的语句,因为它会对记录加 next-key lock,从而避免其他事务插入一条新记录。
第9章 DBA命令
9.1 新建用户
创建一个用户名为java1,密码设置为123的本地用户:
create user 'java1'@'localhost' identified by '123';
创建一个用户名为java2,密码设置为123的外网用户:
create user 'java2'@'%' identified by '123';
采用以上方式新建的用户没有任何权限:系统表也只能看到以下两个
使用root用户查看系统中当前用户有哪些?
select user,host from mysql.user;
9.2 给用户授权
授权语法:grant [权限1,权限2…] on 库名.表名 to ‘用户名’@‘主机名/IP地址’;
给本地用户授权:grant [权限1,权限2…] on 库名.表名 to ‘用户名’@‘localhost’;
给外网用户授权:grant [权限1,权限2…] on 库名.表名 to ‘用户名’@‘%’;
所有权限:all privileges
细粒度权限:select、insert、delete、update、alter、create、drop、index(索引)、usage(登录权限)…
库名可以使用 * ,它代表所有数据库
表名可以采用 * ,它代表所有表
也可以提供具体的数据库和表,例如:powernode.emp (powernode数据库的emp表)
# 将所有库所有表的查询权限赋予本地用户java1
grant select,insert,delete,update,create on *.* to 'java1'@'localhost';
# 将powernode库中所有表的所有权限赋予本地用户java1
grant all privileges on powernode.* to 'java1'@'localhost';
授权后必须刷新权限,才能生效:flush privileges
查看某个用户拥有哪些权限?
show grants for ‘java1’@‘localhost’
show grants for ‘java2’@‘%’
with grant option:
# with grant option的作用是:java2用户也可以给其他用户授权了。
grant select,insert,delete,update on *.* to 'java2'@'%' with grant option;
9.3 撤销用户权限
revoke 权限 on 数据库名.表名 from ‘用户’@‘IP地址’;
# 撤销本地用户java1的insert、update、delete权限
revoke insert, update, delete on powernode.* from 'java1'@'localhost'
# 撤销外网用户java2的insert权限
revoke insert on powernode.* from 'java2'@'%'
撤销权限后也需要刷新权限:flush privileges
注意:撤销权限时 “数据库名.表名” 不能随便写,要求和授权语句时的 “数据库名.表名” 一致。
9.4 修改用户的密码(mysql8版本之后的)
具有管理用户权限的用户才能修改密码,例如root账户可以修改其他账户的密码:
# 本地用户修改密码
alter user 'java1'@'localhost' identified by '456';
# 外网用户修改密码
alter user 'java2'@'%' identified by '456';
修改密码后,也需要刷新权限才能生效:flush privileges
以上是MySQL8版本以后修改用户密码的方式。
9.5 修改用户名
rename user '原始用户名'@'localhost' to '新用户名'@'localhost';
rename user '原始用户名'@'localhost' to '新用户名'@'%';
rename user 'java1'@'localhost' to 'java11'@'localhost';
rename user 'java11'@'localhost' to 'java123'@'%';
flush privileges;
9.6 删除用户
drop user 'java123'@'localhost';
drop user 'java2'@'%';
flush privileges;
9.7 数据备份
- 导出数据(请在登录mysql数据库之前进行)
# 导出powernode这个数据库中所有的表
mysqldump powernode > e:/powernode.sql -uroot -p1234 --default-character-set=utf8
# 导出powernode中emp表的数据
mysqldump powernode emp > e:/powernode.sql -uroot -p1234 --default-character-set=utf8
- 导入数据第一种方式:(请在登录mysql之前进行)
# 现在登录mysql状态下新建一个数据库
create database powernode;
# 在登录mysql之前执行以下命令
mysql powernode < e:/powernode.sql -uroot -p1234 --default-character-set=utf8
- 导入数据第二种方式:(请在登录mysql之后操作)
create database powernode;
use powernode;
source d:/powernode.sql
第10章 MySQL客户端工具
- 对于后端开发人员来说,一个好的MySQL客户端工具可以大大提升开发效率。目前企业中使用最多的是以下三个:
- Navicat for MySQL
2. SQLyog
3. MySQL Workbench
- 安装Navicat for MySQL
- 使用Navicat for MySQL
- 客户端连接MySQL服务器
- 创建数据库(字符集的选择)
- 创建表,设置主键,并且主键自增
- 添加数据(开启事务提交事务)
- 删除数据
- 修改数据
- 导出SQL脚本,导入SQL脚本
- 执行查询(全部执行和选择执行)
- 事务
- 外键
第11章 企业真题
第一题
# 第一步:找小于等于80分的学员姓名
select distinct name from t_student where fenshu <= 80
# 第二步:not in
select distinct name from t_student where name not in(select distinct name from t_student where fenshu <= 80)
第二题
其中,两个表的关联字段为申请单号。
1)查询身份证号为440401430103082的申请日期。
2)查询同一个身份证号码有两条以上记录的身份证号码及记录个数。
3)将身份证号码为440401430103082的记录在两个表中的申请状态均改为07。
4)删除g_cardapplydetail表中所有姓李的记录。
模拟数据:考试做这种题目最重要的是要冷静下来,只有静下来SQL才能写好。要模拟数据。看到数据SQL就好写了。
1)查询身份证号为440401430103082的申请日期。
bigint转date,可以使用from_unixtime函数。
select a.g_applydate from g_cardapply a join g_cardapplydetail b on a.g_applyno = b.g_applyno where b.g_idcard = '440401430103082'
2)查询同一个身份证号码有两条以上记录的身份证号码及记录个数。
select count(g_idcard),g_idcard from g_cardapplydetail group by g_idcard having count(g_idcard) >= 2
3)将身份证号码为440401430103082的记录在两个表中的申请状态均改为07。
UPDATE
g_cardapply
JOIN
g_cardapplydetail
ON
g_cardapply.g_applyno = g_cardapplydetail.g_applyno
AND
g_cardapplydetail.g_idcard = '440401430103082'
SET g_cardapply.g_state = '07',
g_cardapplydetail.g_state = '07'
4)删除g_cardapplydetail表中所有姓李的记录。
delete t1,t2 from g_cardapply t1 join g_cardapplydetail t2 on t1.g_applyno=t2.g_applyno where t2.g_name like '李%';
第三题
表名:stuscore
1)统计如下:课程不及格[0-59]的多少个,良[60-80]多少个,优[81-100]多少个。
2)计算科科及格的人的平均成绩。
第四题
1)请用一条SQL语句查询出不同部门中担任“钳工”的职工平均工资。
2)请用一条SQL语句查询出不同部门中担任“钳工”的职工平均工资高于2000的部门。
第五题
Employee是雇员信息表:
雇员姓名(主键):person-name
街道:street
城市:city
Company是公司信息表:
公司名称(主键):company-name
城市:city
Works是雇员工作信息表:
雇员姓名(主键):person-name
公司名称:company-name
年薪:salary
Manages是雇员工作关系表:
雇员姓名(主键):person-name
经理姓名:manager-name
模拟数据:
员工表:employee
公司表:company
雇员工作信息表:Works
雇员工作关系表:Manages
请给出下面每一个查询的SQL语句:
- 找出所有居住地与工作的公司在同一城市的员工的姓名。
- 找出比Small Bank Corporation的所有员工收入都高的所有员工的姓名。
- 找出平均年薪在10000美元以上的公司及其平均年薪。
第六题
客户表Client
订单表Order
客户订单表ClientOrder
图书表Book
- 请写出一条SQL语句,查询出每个客户的所有订单并按照地址排序,要求输出格式为:address client_name phone order_id
- 请写出一条SQL语句,查询出每个客户订购的图书总价。要求输出格式为:client_name total_price
- 如果要求每个订单可以包含多种图书,应该如何修改Order表的主键?为了保证每个订单只被一个客户拥有,应该在ClientOrder表上增加怎样的约束?
第七题
模拟数据:
学生表:student
课程表:course
成绩表:sc
教师表:teacher
- 查询1号课比2号课成绩高的所有学生学号。
- 查询平均成绩大于60分的学号和平均成绩。
- 查询所有学生学号、姓名、选课数、总成绩。
- 查询姓“李”的老师的个数。
- 查询没学过“叶平”老师课的学号、姓名。
第八题
学生表:student
课程表:class
选课表:chosen_class
- 没有选修课程编号为C1的学生姓名
- 列出每门课程名称和平均成绩,并按照成绩排序
- 选了2门课以上的学生姓名。
第九题
要转换成:
MySQL行转列
MySQL行转列又叫做数据透视。什么叫做行转列?将原本横向排列的数据透视成纵向排列的数据,进而进行计算、分析、展示等操作。
假设有一个学生选课成绩表,包含学生姓名(stu_name)、课程名称(course_name)和分数(score)三个字段。在原始数据中,每个学生在不同的课程中都有自己的得分情况,数据样例如下:
stu_name | course_name | score |
---|---|---|
张三 | 数学 | 80 |
张三 | 英语 | 85 |
张三 | 历史 | 90 |
李四 | 数学 | 75 |
李四 | 英语 | 92 |
李四 | 历史 | 85 |
王五 | 数学 | 88 |
王五 | 英语 | 90 |
王五 | 历史 | 95 |
可以使用行转列操作,将每个学生在不同课程中的分数拆分成多条记录,每条记录包含一个课程以及对应的分数。转换后的数据样例如下:
stu_name | 数学 | 英语 | 历史 |
---|---|---|---|
张三 | 80 | 85 | 90 |
李四 | 75 | 92 | 85 |
王五 | 88 | 90 | 95 |
从上表中可以看出,在行转列之后,每一行记录都表示了一个学生在不同课程中的分数。这样更便于对不同科目的分数进行比较、计算平均值等分析操作。
使用case when+group by完成
drop table if exists t_student;
create table t_student(
stu_name varchar(10),
course_name varchar(10),
score int
);
insert into t_student(stu_name, course_name, score) values('张三', '数学', 80);
insert into t_student(stu_name, course_name, score) values('张三', '英语', 85);
insert into t_student(stu_name, course_name, score) values('张三', '历史', 90);
insert into t_student(stu_name, course_name, score) values('李四', '数学', 75);
insert into t_student(stu_name, course_name, score) values('李四', '英语', 92);
insert into t_student(stu_name, course_name, score) values('李四', '历史', 85);
insert into t_student(stu_name, course_name, score) values('王五', '数学', 88);
insert into t_student(stu_name, course_name, score) values('王五', '英语', 90);
insert into t_student(stu_name, course_name, score) values('王五', '历史', 95);
commit;
select * from t_student;
行转列后的效果是:
sql如下:
select
stu_name,
max(case course_name when '数学' then score else 0 end) as '数学',
max(case course_name when '英语' then score else 0 end) as '英语',
max(case course_name when '历史' then score else 0 end) as '历史'
from
t_student
group by
stu_name;
通过以上内容的学习,我们这个面试题就迎刃而解了:
select
year,
max(case season when '一季度' then count else 0 end) as '一季度',
max(case season when '二季度' then count else 0 end) as '二季度',
max(case season when '三季度' then count else 0 end) as '三季度',
max(case season when '四季度' then count else 0 end) as '四季度'
from
t_temp
group by
year;
第十题
select
x.a 开始数字, y.a 结束数字
from
(select m.a,row_number() over(order by m.a) as rownum from (select a, lag(a) over(order by a asc) as pre_a from t) m where m.a - m.pre_a != 1 or m.pre_a is null) x
join
(select n.a,row_number() over(order by n.a) as rownum from (select a, lead(a) over(order by a asc) as next_a from t) n where n.next_a - n.a != 1 or n.next_a is null) y
on
x.rownum = y.rownum;
解答上面这个题目需要具备以下知识点:
- lag函数
- lead函数
- row_number函数
lag函数:获取当前行的上一行数据
select empno,ename,sal,(lag(sal) over(order by sal asc)) as pre_sal from emp;
注意:over函数用来指定“在…范围内”,通常和lag函数联用。
lead函数:获取当前行的下一行数据
select empno,ename,sal,(lead(sal) over(order by sal asc)) as next_sal from emp;
注意:over函数用来指定“在…范围内”,通常和lead函数联用。
row_number函数:可以为查询结果集生成行号:
select empno,ename,sal,row_number() over(order by sal) as rownum from emp;
利用row_number函数,将两个不相关的列拼接在一起显示:
select
x.a, y.b
from
(select a,row_number() over(order by a) as rownum from t1) x
join
(select b,row_number() over(order by b) as rownum from t2) y
on
x.rownum = y.rownum;
CTE语法(公用表表达式):Common Table Expression。创建临时表的一种语法:
-- 查询每个部门平均工资的工资等级
-- 第一种写法
select
t.deptno,t.avgsal,s.grade
from
(select deptno,avg(sal) as avgsal from emp group by deptno) t
join
salgrade s
on
t.avgsal between s.losal and s.hisal;
-- 第二种写法:使用CTE语法
with cte_exp as(select deptno,avg(sal) as avgsal from emp group by deptno)
select
cte_exp.deptno,cte_exp.avgsal,s.grade
from
cte_exp
join
salgrade s
on
cte_exp.avgsal between s.losal and s.hisal;
partition by:将数据分区,和group by区别是:group by是分组,然后和分组函数一起用。partition by分区不需要和分组函数一起使用
select deptno, empno,ename,sal,(lag(sal) over(partition by deptno order by sal asc)) as pre_sal from emp;
MySQL 8.0及以上版本中支持如下常用的窗口函数:
- ROW_NUMBER():排名函数,返回当前结果集中每个行的行号;
- RANK():排名函数,计算分组结果中的排名,相同的行排名相同且没有空缺,下一个行排名跳过空缺;
- DENSE_RANK():排名函数,计算分组结果中的排名,相同的行排名相同,排名连续,没有空缺;
- NTILE():将分组结果等分为指定的组数,计算每组的大小;
- LAG():返回分组内前一行的值;
- LEAD():返回分组内后一行的值;
- FIRST_VALUE():返回分组内第一个值;
- LAST_VALUE():返回分组内最后一个值;
- AVG()、SUM()、COUNT()、MIN()、MAX():聚合函数,可以配合OVER()进行窗口操作。
需要注意的是,MySQL的窗口函数和其他DBMS中的窗口函数相比较,可能略有不同,需要根据MySQL的文档进行使用。