目录
1、SQL语句执行流程
MySQL大体上可分为Server层
和存储引擎层
两部分。
Server层:
-
连接器
:TCP握手后服务器来验证登陆用户身份,A用户创建连接后,管理员对A用户权限修改了也不会影响到已经创建的链接权限,必须重新登陆。 -
查询缓存
:查询后的结果存储位置,MySQL8.0版本以后已经取消,因为查询缓存失效太频繁,得不偿失。 -
分析器
:根据语法规则,判断你输入的这个SQL语句是否满足MySQL语法。 -
优化器
:多种执行策略可实现目标,系统自动选择最优进行执行。 -
执行器
:判断是否有权限,将最终任务提交到存储引擎。
存储引擎层
负责数据的存储和提取。其架构模式是插件式
的,支持InnoDB
、MyISAM
、Memory
等多个存储引擎。现在最常用的存储引擎是InnoDB
,它从MySQL 5.5.5版本开始成为了默认存储引擎(经常用的也是这个)。
SQL执行顺序
2、BinLog、RedoLog、UndoLog
BinLog
BinLog
是记录所有数据库表结构变更(例如create、alter table)以及表数据修改(insert、update、delete)的二进制日志,主从数据库同步用到的都是BinLog文件。BinLog日志文件有三种模式。
STATEMENT 模式
ROW 模式
MIXED 模式
主从同步流程:
mysql默认的复制方式是异步
的,并且复制的时候是有并行复制能力
的。主库把日志发送给从库后不管了,这样会产生一个问题就是假设主库挂了,从库处理失败了,这时候从库升为主库后,日志就丢失了。由此产生两个概念。
-
全同步复制
-
半同步复制
还可以延伸到由于主从配置不一样、主库大事务、从库压力过大、网络震荡等造成主备延迟
,如何避免这个问题?主备切换的时候用可靠性优先原则
还是可用性优先原则
?如何判断主库Crash了?互为主备情况下如何避免主备循环复制?被删库跑路了如何正确恢复?(⊙o⊙)… 感觉越来越扯到DBA的活儿上去了。
RedoLog
可以先通过下面demo理解:
饭点记账可以把账单写在账本
上也可以写在粉板
上。有人赊账或者还账的话,一般有两种做法:
生意忙时选后者,因为前者太麻烦了。得在密密麻麻的记录中找到这个人的赊账总额信息,找到之后再拿出算盘计算,最后再将结果写回到账本上。
同样在MySQL中如果每一次的更新操作都需要写进磁盘,然后磁盘也要找到对应的那条记录,然后再更新,整个过程IO成本、查找成本都很高。而粉板和账本配合的整个过程就是MySQL用到的是Write-Ahead Logging 技术,它的关键点就是先写日志,再写磁盘
。此时账本 = BinLog,粉板 = RedoLog。
RedoLog有write pos
跟checkpoint
write pos和check point之间的是粉板上还空着的部分,可以用来记录新的操作。如果write pos追上checkpoint,表示粉板满了,这时候不能再执行新的更新,得停下来先擦掉一些记录,把checkpoint推进一下。
有了redo log,InnoDB就可以保证即使数据库发生异常重启,之前提交的记录都不会丢失,这个能力称为crash-safe
。
redolog两阶段提交
:为了让binlog跟redolog两份日志之间的逻辑一致。提交流程大致如下:
-
当在2之前崩溃时,重启恢复后发现没有commit,回滚。备份恢复:没有binlog 。一致
-
当在3之前崩溃时,重启恢复发现虽没有commit,但满足prepare和binlog完整,所以重启后会
自动
commit。备份:有binlog. 一致
binlog跟redolog区别:
-
redo log是InnoDB引擎特有的;binlog是MySQL的Server层实现的,所有引擎都可以使用。
-
redo log是物理日志,记录的是在某个数据页上做了什么修改;binlog是逻辑日志,记录的是这个语句的原始逻辑,比如给ID=2这一行的c字段加1。
-
redo log是循环写的,空间固定会用完;binlog是可以追加写入的。追加写是指binlog文件写到一定大小后会切换到下一个,并不会覆盖以前的日志。
UndoLog
UndoLog 一般是逻辑日志,主要分为两种:
-
insert undo log
-
update undo log
3、MySQL中的索引
索引的常见模型有哈希表
、有序数组
和搜索树
。
B+树比B树优势
在于:
索引的优点:
索引的缺点:
索引设计的原则不应该:
应该:
索引失效的场景
关于索引的知识点:
主键索引
:主键索引的叶子节点存的是整行
数据信息。在InnoDB里,主键索引也被称为聚簇索引(clustered index)。主键自增是无法保证完全自增的哦
,遇到唯一键冲突、事务回滚等都可能导致不连续。
唯一索引
:以唯一列生成的索引,该列不允许有重复值,但允许有空值(NULL)
普通索引跟唯一索引查询性能
:InnoDB的数据是按数据页为单位来读写的,默认每页16KB,因此这两种索引查询数据性能差别微乎其微。
change buffer
:普通索引用在更新过程的加速,更新的字段如果在缓存中,如果是普通索引则直接更新即可。如果是唯一索引需要将所有数据读入内存来确保不违背唯一性,所以尽量用普通索引。
非主键索引
:非主键索引的叶子节点内容是主键
的值。在InnoDB里,非主键索引也被称为二级索引(secondary index)
回表
:先通过数据库索引扫描出数据所在的行,再通过行主键id取出索引中未提供的数据,即基于非主键索引的查询需要多扫描一棵索引树。
覆盖索引
:如果一个索引包含(或者说覆盖)所有需要查询的字段的值,我们就称之为覆盖索引。
联合索引
:相对单列索引,组合索引是用多个列组合构建的索引,一次性最多联合16个。
最左前缀原则
:对多个字段同时建立的组合索引(有顺序,ABC,ACB是完全不同的两种联合索引) 以联合索引(a,b,c)为例,建立这样的索引相当于建立了索引a、ab、abc三个索引。另外组合索引实际还是一个索引,并非真的创建了多个索引,只是产生的效果等价于产生多个索引。
索引下推
:MySQL 5.6引入了索引下推优化,可以在索引遍历过程中,对索引中包含的字段先做判断,过滤掉不符合条件的记录,减少回表字数。
索引维护
:B+树为了维护索引有序性涉及到页分裂跟页合并。增删数据时需考虑页空间利用率。
自增主键
:一般会建立与业务无关的自增主键,不会触发叶子节点分裂。
延迟关联
:通过使用覆盖索引查询返回需要的主键,再根据主键关联原表获得需要的数据。
InnoDB存储
: * .frm
文件是一份定义文件,也就是定义数据库表是一张怎么样的表。*.ibd
文件则是该表的索引,数据存储文件,既该表的所有索引树,所有行记录数据都存储在该文件中。
MyISAM存储
:* .frm
文件是一份定义文件,也就是定义数据库表是一张怎么样的表。* .MYD
文件是MyISAM存储引擎表的所有行数据的文件。* .MYI
文件存放的是MyISAM存储引擎表的索引相关数据的文件。MyISAM引擎下,表数据和表索引数据是分开存储的。
MyISAM查询
:在MyISAM下,主键索引和辅助键索引都属于非聚簇索引。查询不管是走主键索引,还是非主键索引,在叶子结点得到的都是目的数据的地址,还需要通过该地址,才能在数据文件中找到目的数据。
PS
:InnoDB支持聚簇索引,MyISAM不支持聚簇索引
4、SQL事务隔离级别
ACID的四个特性
-
原子性
(Atomicity):把多个操作放到一个事务中,保证这些操作要么都成功,要么都不成功 -
一致性
(Consistency):理解成一串对数据进行操作的程序执行下来,不会对数据产生不好的影响,比如凭空产生,或消失 -
隔离性
(Isolation,又称独立性):隔离性的意思就是多个事务之间互相不干扰,即使是并发事务的情况下,他们只是两个并发执行没有交集,互不影响的东西;当然实现中,也不一定需要这么完整隔离性,即不一定需要这么的互不干扰,有时候还是允许有部分干扰的。所以MySQL可以支持4种事务隔离性 -
持久性
(Durability):当某个操作操作完毕了,那么结果就是这样了,并且这个操作会持久化到日志记录中
PS:ACID中C与CAP定理中C的区别
事务操作可能会出现的数据问题
在说隔离级别之前,你首先要知道,你隔离得越严实,效率就会越低
。因此很多时候,我们都要在二者之间寻找一个平衡点。SQL标准的事务隔离级别由低到高如下:
上图从上到下的模式会导致系统的并行性能依次降低,安全性依次提高。
标准跟实现
:上面都是关于事务的标准,但是每一种数据库都有不同的实现,比如MySQL InnDB
默认为RR
级别,但是不会出现幻读。因为当事务A更新了所有记录的某个字段,此时事务A会获得对这个表的表锁,因为事务A还没有提交,所以事务A获得的锁没有释放,此时事务B在该表插入新记录,会因为无法获得该表的锁,则导致插入操作被阻塞。只有事务A提交了事务后,释放了锁,事务B才能进行接下去的操作。所以可以说 MySQL的RR级别的隔离是已经实现解决了脏读,不可重复读和幻读的。
5、MySQL中的锁
无论是Java的并发编程还是数据库的并发操作都会涉及到锁,研发人员引入了悲观锁
跟乐观锁
这样一种锁的设计思想。
悲观锁:
乐观锁:
数据库并发场景主要有三种:
两类更新丢失问题:
为了合理贯彻落实锁的思想,MySQL中引入了杂七杂八的各种锁:
锁分类
MySQL支持三种层级的锁定,分别为
-
表级锁定
-
页级锁定
-
行级锁定
MyISAM中的锁
-
虽然MySQL支持表,页,行三级锁定,但MyISAM存储引擎只支持表锁。所以MyISAM的加锁相对比较开销低,但数据操作的并发性能相对就不高。但如果写操作都是尾插入,那还是可以支持一定程度的读写并发
-
从MyISAM所支持的锁中也可以看出,MyISAM是一个支持读读并发,但不支持通用读写并发,写写并发的数据库引擎,所以它更适合用于读多写少的应用场合,一般工程中也用的较少。
InnoDB中的锁
该模式下支持的锁实在是太多了,具体如下:
举个栗子,比如行锁里的共享锁跟排它锁:lock in share modle
共享读锁:
for update
排它写锁:
Gap Lock
间隙锁:
间隙锁加锁原则(学完忘那种):
6、MVCC
MVCC:
MySQL InnoDB下的当前读和快照读
-
当前读
-
快照读
因为大佬不满意只让数据库采用悲观锁这样性能不佳的形式去解决读-写冲突问题,而提出了MVCC,所以我们可以形成两个组合:
MVCC的实现原理
MVCC实现原理主要是依赖记录中的 四个隐式字段
、undo日志
、Consistent Read View
来实现的。
四个隐式字段:
-
DB_TRX_ID:
-
DB_ROLL_PTR
-
DB_ROW_ID
-
FLAG
事务对一条记录的修改,会导致该记录的undo log成为一条记录版本线性表(链表
),undo log的链首就是最新的旧记录,链尾就是最早的旧记录。
undo日志:此知识点上文已经说过了,对MVCC有帮助的实质是update undo log,undo log实际上就是存在rollback segment中旧记录链。
一致读视图 Consistent Read View:Read View是事务进行快照读操作的时候生产的读视图(Read View),在该事务执行的快照读的那一刻,会生成数据库系统当前的一个快照
,记录并维护系统当前活跃事务的ID(InnoDB里面每个事务有一个唯一的事务ID,叫作transaction id
。它是在事务开始的时候向InnoDB的事务系统申请的,是按申请顺序严格递增的)。拿着这个ID跟记录中ID对比进行选择性展示,这里说下大致的思维
。
你可以简单的理解为MVCC为每一行增加了两个隐藏字段,两个字段分别保存了这个行的当前事务ID
跟行的删除事务ID
。
-
insert时:
-
select时:
-
delete时:
-
update时:
上面只是一个浅显的讲解MVCC选择标准流程,源码层面应该是根据低水位
跟高水位
来截取的。具体实现可自行百度。
重点
:
7、缓冲池(buffer pool)
应用系统分层架构,为了加速数据访问,会把最常访问的数据,放在缓存(cache)里,避免每次都去访问数据库。操作系统,会有缓冲池(buffer pool)机制,避免每次访问磁盘,以加速数据的访问。MySQL作为一个存储系统,同样具有缓冲池(buffer pool)机制,以避免每次查询数据都进行磁盘IO,主要作用:
预读失效:
缓冲池污染:
8、table瘦身
空洞:
重建表思路:
重建表指令:
9、SQL Joins、统计、 随机查询
7种join具体如下:
统计:
随机查询:
mysql> select word from words order by rand() limit 3;
直接使用order by rand()
,explain 这个语句发现需要 Using temporary
和 Using filesort
,查询的执行代价往往是比较大的。所以在设计的时要避开这种写法。
mysql> select count(*) into @C from t;
set @Y1 = floor(@C * rand());
set @Y2 = floor(@C * rand());
set @Y3 = floor(@C * rand());
select * from t limit @Y1,1;
select * from t limit @Y2,1;
select * from t limit @Y3,1;
这样可以避免临时表跟排序的产生,最终查询行数 = C + (Y1+1) + (Y2+1) + (Y3+1)
exist 和 in 对比:
10、MySQL优化
SQL优化主要分4个方向:SQL语句跟索引
、表结构
、系统配置
、硬件
。
总优化思路就是最大化利用索引、尽可能避免全表扫描、减少无效数据的查询:
SQL语句优化大致举例:
SQL调优大致思路:
1、先用慢查询日志定位具体需要优化的sql
2、使用 explain 执行计划查看索引使用情况
3、重点关注(一般情况下根据这4列就能找到索引问题):
4、根据上1步找出的索引问题优化sql 5、再回到第2步
表结构优化:
读写分离:
分库分表:分库分表分为垂直和水平两个方式,一般是先垂直后水平
。
目前主要流行的分库分表工具 就是Mycat
和sharding-sphere
。
TiDB:开源分布式
数据库,结合了传统的 RDBMS 和NoSQL 的最佳特性。TiDB 兼容 MySQL,支持无限的水平扩展
,具备强一致性和高可用性。TiDB 的目标是为 OLTP(Online Transactional Processing) 和 OLAP (Online Analytical Processing) 场景提供一站式的解决方案。TiDB 具备如下核心特点
适合:
不适合: