PS:本文已经收录到github仓库,此仓库用于分享Java核心知识,包括Java基础、MySQL、SpringBoot、Mybatis、Redis、RabbitMQ等等,面试必备。
github地址:https://github.com/Tyson0314/Java-learning
如果github访问不了,可以访问gitee仓库。
gitee地址:https://gitee.com/tysondai/Java-learning
事务特性:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。
先了解下几个概念:脏读、不可重复读、幻读。
不可重复读和脏读的区别是,脏读是某一事务读取了另一个事务未提交的脏数据,而不可重复读则是读取了前一事务提交的数据。
幻读和不可重复读都是读取了另一条已经提交的事务,不同的是不可重复读的重点是修改,幻读的重点在于新增或者删除。
事务隔离就是为了解决上面提到的脏读、不可重复读、幻读这几个问题。
MySQL数据库为我们提供的四种隔离级别:
查看隔离级别:
select @@transaction_isolation;
设置隔离级别:
set session transaction isolation level read uncommitted;
索引是存储引擎用于提高数据库表的访问速度的一种数据结构。
特点:1、避免进行数据库全表的扫描,大多数情况,只需要扫描较少的索引页和数据页;提升查询语句的执行效率,但降低了新增、删除操作的速度,同时也会占用额外的存储空间。
数据是存储在磁盘上的,查询数据时,如果没有索引,会加载所有的数据到内存,依次进行检索,读取磁盘次数较多。有了索引,就不需要加载所有数据,因为B+树的高度一般在2-4层,最多只需要读取2-4次磁盘,查询速度大大提升。
什么情况下需要建索引:
什么情况下不建索引?
B+ 树是基于B 树和叶子节点顺序访问指针进行实现,它具有B树的平衡性,并且通过顺序访问指针来提高区间查询的性能。
在 B+ 树中,节点中的 key 从左到右递增排列,如果某个指针的左右相邻 key 分别是 keyi 和 keyi+1,则该指针指向节点的所有 key 大于等于 keyi 且小于等于 keyi+1。
进行查找操作时,首先在根节点进行二分查找,找到key所在的指针,然后递归地在指针所指向的节点进行查找。直到查找到叶子节点,然后在叶子节点上进行二分查找,找出 key 所对应的数据项。
MySQL 数据库使用最多的索引类型是BTREE索引,底层基于B+树数据结构来实现。
mysql> show index from blog\G;
*************************** 1. row ***************************
Table: blog
Non_unique: 0
Key_name: PRIMARY
Seq_in_index: 1
Column_name: blog_id
Collation: A
Cardinality: 4
Sub_part: NULL
Packed: NULL
Null:
Index_type: BTREE
Comment:
Index_comment:
Visible: YES
Expression: NULL
下面来看看一个索引的例子:
如下图,col1 是主键,col2和col3是普通字段。
下图是主键索引对应的 B+树结构,每个节点对应磁盘的一页。
对col3 建立一个单列索引,对应的B+树结构:
主键索引:名为primary的唯一非空索引,不允许有空值。
唯一索引:索引列中的值必须是唯一的,但是允许为空值。
唯一索引和主键索引的区别是:UNIQUE 约束的列可以为null且可以存在多个null值。UNIQUE KEY的用途:唯一标识数据库表中的每条记录,主要是用来防止数据重复插入。
创建唯一索引:
ALTER TABLE table_name
ADD CONSTRAINT constraint_name UNIQUE KEY(column_1,column_2,...);
组合索引:在表中的多个字段组合上创建的索引,只有在查询条件中使用了这些字段的左边字段时,索引才会被使用,使用组合索引时遵循最左前缀原则。
全文索引:全文索引,只有在MyISAM引擎上才能使用,只能在CHAR,VARCHAR,TEXT类型字段上使用全文索引。
如果 SQL 语句中用到了组合索引中的最左边的索引,那么这条 SQL 语句就可以利用这个组合索引去进行匹配。当遇到范围查询(>、<、between、like)就会停止匹配,后面的字段不会用到索引。
对(a,b,c)建立索引,查询条件使用 a/ab/abc 会走索引,使用 bc 不会走索引。
对(a,b,c,d)建立索引,查询条件为a = 1 and b = 2 and c > 3 and d = 4
,那么,a,b,c三个字段能用到索引,而d就匹配不到。因为遇到了范围查询!
对(a, b) 建立索引,a 在索引树中是全局有序的,而 b 是全局无序,局部有序(当a相等时,会对b进行比较排序)。直接执行b = 2
这种查询条件没有办法利用索引。
从局部来看,当a的值确定的时候,b是有序的。例如a = 1时,b值为1,2是有序的状态。当a=2时候,b的值为1,4也是有序状态。 因此,你执行a = 1 and b = 2
是a,b字段能用到索引的。而你执行a > 1 and b = 2
时,a字段能用到索引,b字段用不到索引。因为a的值此时是一个范围,不是固定的,在这个范围内b值不是有序的,因此b字段用不上索引。
InnoDB使用表的主键构造主键索引树,同时叶子节点中存放的即为整张表的记录数据。聚集索引叶子节点的存储是逻辑上连续的,使用双向链表连接,叶子节点按照主键的顺序排序,因此对于主键的排序查找和范围查找速度比较快。
聚集索引的叶子节点就是整张表的行记录。InnoDB 主键使用的是聚簇索引。聚集索引要比非聚集索引查询效率高很多。
对于InnoDB来说,聚集索引一般是表中的主键索引,如果表中没有显示指定主键,则会选择表中的第一个不允许为NULL的唯一索引。如果没有主键也没有合适的唯一索引,那么innodb内部会生成一个隐藏的主键作为聚集索引,这个隐藏的主键长度为6个字节,它的值会随着数据的插入自增。
select的数据列只用从索引中就能够取得,不需要到数据表进行二次查询,换句话说查询列要被所使用的索引覆盖。
对于innodb表的二级索引,如果索引能覆盖到查询的列,那么就可以避免对主键索引的二次查询。
不是所有类型的索引都可以成为覆盖索引。覆盖索引要存储索引列的值,而哈希索引、全文索引不存储索引列的值,所以MySQL只能使用b+树索引做覆盖索引。
对于使用了覆盖索引的查询,在查询前面使用explain,输出的extra列会显示为using index。
比如user_like 用户点赞表,组合索引为(user_id, blog_id),user_id和blog_id都不为null。
explain select blog_id from user_like where user_id = 13;
Extra中为Using index
,查询的列被索引覆盖,并且where筛选条件符合最左前缀原则,通过索引查找就能直接找到符合条件的数据,不需要回表查询数据。
explain select user_id from user_like where blog_id = 1;
Extra中为Using where; Using index
, 查询的列被索引覆盖,where筛选条件不符合最左前缀原则,无法通过索引查找找到符合条件的数据,但可以通过索引扫描找到符合条件的数据,也不需要回表查询数据。
explain select blog_id from user_like where status = 1;
Extra中为Using where
,查询时未找到可用的索引,进而通过where
条件过滤获取所需数据。
%abc
,无法使用索引;非%开头的like查询如abc%
,相当于范围查询,会使用索引MySQL 5.5版本后默认的存储引擎为InnoDB。
InnoDB是MySQL默认的事务型存储引擎,使用最广泛,基于聚簇索引建立的。InnoDB内部做了很多优化,如能够自动在内存中创建自适应hash索引,以加速读操作。
优点:支持事务和崩溃修复能力。InnoDB引入了行级锁和外键约束。
缺点:占用的数据空间相对较大。
适用场景:需要事务支持,并且有较高的并发读写频率。
数据以紧密格式存储。对于只读数据,或者表比较小、可以容忍修复操作,可以使用MyISAM引擎。MyISAM会将表存储在两个文件中,数据文件.MYD和索引文件.MYI。
优点:访问速度快。
缺点:MyISAM不支持事务和行级锁,不支持崩溃后的安全恢复,也不支持外键。
适用场景:对事务完整性没有要求;只读的数据,或者表比较小,可以忍受修复repair操作。
MyISAM特性:
CHECK TABLE tablename
检查表的错误,如果有错误执行REPAIR TABLE tablename
进行修复。MEMORY引擎将数据全部放在内存中,访问速度较快,但是一旦系统奔溃的话,数据都会丢失。
MEMORY引擎默认使用哈希索引,将键的哈希值和指向数据行的指针保存在哈希索引中。哈希索引使用拉链法来处理哈希冲突。
优点:访问速度较快。
缺点:
是否支持行级锁 : MyISAM 只有表级锁,而InnoDB 支持行级锁和表级锁,默认为行级锁。
是否支持事务和崩溃后的安全恢复: MyISAM 强调的是性能,每次查询具有原子性,其执行速度比InnoDB类型更快,但是不提供事务支持。但是InnoDB 提供事务支持,具有事务、回滚和崩溃修复能力。
是否支持外键: MyISAM不支持,而InnoDB支持。
是否支持MVCC :仅 InnoDB 支持。应对高并发事务,MVCC比单纯的加锁更高效;MVCC只在 READ COMMITTED
和 REPEATABLE READ
两个隔离级别下工作;MVCC可以使用乐观锁和悲观锁来实现;各数据库中MVCC实现并不统一。
MyISAM不支持聚集索引,InnoDB支持聚集索引。
myisam引擎主键索引和其他索引区别不大,叶子节点都包含索引值和行指针。
innodb引擎二级索引叶子存储的是索引值和主键值(不是行指针),这样可以减少行移动和数据页分裂时二级索引的维护工作。
MVCC(Multiversion concurrency control
) 就是同一份数据保留多版本的一种方式,进而实现并发控制。可以认为MVCC是行级锁的变种。在查询的时候,通过read view和版本链找到对应版本的数据。
MVCC只适用于read committed和repeatable read。使用事务更新行记录时,会生成一个新的版本的行记录。
作用:提升并发性能。对于高并发场景,MVCC比行级锁更有效、开销更小。
mvcc实现依赖于版本链,版本链是通过表的三个隐藏字段实现。
事务id:data_trx_id,当前事务id
回滚指针:data_roll_ptr,指向当前行记录的上一个版本,通过这个指针将数据的多个版本连接在一起构成undo log版本链
主键:db_row_id,如果数据表没有主键,InnoDB会自动生成主键
使用事务更新行记录的时候,就会生成版本链:
read view就是在某一时刻给事务打snapshot快照。在read_view内部维护一个活跃事务链表,这个链表包含在创建read view之前还未提交的事务,不包含创建read view之后提交的事务。
不同隔离级别创建read view的时机不同。
read committed:每次执行select都会创建新的read_view,保证能读取到其他事务已经提交的修改。
repeatable read:在一个事务范围内,第一次select时更新这个read_view,以后不会再更新,后续所有的select都是复用之前的read_view。这样可以保证事务范围内每次读取的内容都一样,即可重复读。
当访问数据行时,会先判断当前版本数据项是否可见,如果是不可见的,会通过版本链找到一个可见的版本。
总结:通过比较read view和数据行的当前版本,找到当前事务可见的版本,进而实现read commit和repeatable read的事务隔离级别。
记录的两种读取方式。
快照读:读取的是快照版本,也就是历史版本。普通的SELECT就是快照读。通过MVCC来进行控制的,不用加锁。
当前读:读取的是最新版本。UPDATE、DELETE、INSERT、SELECT … LOCK IN SHARE MODE、SELECT … FOR UPDATE是当前读。
快照读情况下,InnoDB通过mvcc机制避免了幻读现象。而mvcc机制无法避免当前读情况下出现的幻读现象。
事务a和事务b同时开启事务,事务a插入数据然后提交,事务b执行全表的update,然后执行查询,查到了事务A中添加的数据。
MySQL如何实现避免幻读:
next-key包括两部分:行锁和间隙锁。行锁是加在索引上的锁,间隙锁是加在索引之间的。
select * from table where id<6 lock in share mode;--共享锁 锁定的是小于6的行和等于6的行
select * from table where id<6 for update;--排他锁
实际上很多的项目中是不会使用到上面的两种方法的,串行化读的性能太差,而且其实幻读很多时候是我们完全可以接受的。
Serializable隔离级别也可以避免幻读,会锁住整张表,并发性极低,一般很少使用。
在SELECT 的读取锁定主要分为两种方式:共享锁和排他锁。
SELECT ... LOCK IN SHARE MODE
SELECT ... FOR UPDATE
这两种方式主要的不同在于LOCK IN SHARE MODE 多个事务同时更新同一个表单时很容易造成死锁。这种情况最好使用SELECT …FOR UPDATE。
select * from goods where id = 1 for update
:申请排他锁的前提是,没有线程对该结果集的任何行数据使用排它锁或者共享锁,否则申请会受到阻塞。在进行事务操作时,MySQL会对查询结果集的每行数据添加排它锁,其他线程对这些数据的更改或删除操作会被阻塞(只能读操作),直到该语句的事务被commit语句或rollback语句结束为止。
select… for update 使用注意事项
当单表的数据量达到1000W或100G以后,优化索引、添加从库等可能对数据库性能提升效果不明显,此时就要考虑对其进行切分了。切分的目的就在于减少数据库的负担,缩短查询的时间。
数据切分可以分为两种方式:垂直(纵向)划分和水平(横向)划分。
垂直划分数据库是根据业务进行划分,例如将shop库中涉及商品、订单、用户的表分别划分出成一个库,通过降低单库的大小来提高性能,但这种方式并没有解决高数据量带来的性能损耗。同样的,分表的情况就是将一个大表根据业务功能拆分成一个个子表,例如商品基本信息和商品描述,商品基本信息一般会展示在商品列表,商品描述在商品详情页,可以将商品基本信息和商品描述拆分成两张表。
优点:行记录变小,数据页可以存放更多记录,在查询时减少I/O次数。
缺点:
水平划分是根据一定规则,例如时间或id序列值等进行数据的拆分。比如根据年份来拆分不同的数据库。每个数据库结构一致,但是数据得以拆分,从而提升性能。
优点:单库(表)的数据量得以减少,提高性能;切分出的表结构相同,程序改动较少。
缺点:
MySQL日志 主要包括查询日志、慢查询日志、事务日志、错误日志、二进制日志等。其中比较重要的是二进制日志binlog和事务日志 redo log(重做日志)和 undo log(回滚日志)。
二进制日志(bin log)是MySQL数据库级别的文件,记录对MySQL数据库执行修改的所有操作,不会记录select和show语句,主要用于恢复数据库和同步数据库。
查看bin log是否开启,以及保存位置:
MySQL> show variables like '%log_bin%';
+---------------------------------+----------------------------------------------------+
| Variable_name | Value |
+---------------------------------+----------------------------------------------------+
| log_bin | ON |
| log_bin_basename | F:\java\MySQL8\data\Data\DESKTOP-8F30VS1-bin |
| log_bin_index | F:\java\MySQL8\data\Data\DESKTOP-8F30VS1-bin.index |
| log_bin_trust_function_creators | OFF |
| log_bin_use_v1_row_events | OFF |
| sql_log_bin | ON |
+---------------------------------+----------------------------------------------------+
关闭bin log,找到/etc/my.cnf文件,注释以下代码:
log-bin=MySQL-bin
binlog_format=mixed
重做日志(redo log)是Innodb引擎级别,用来记录Innodb存储引擎的事务日志,不管事务是否提交都会记录下来,用于数据恢复。当数据库发生故障,InnoDB存储引擎会使用redo log恢复到发生故障前的时刻,以此来保证数据的完整性。将参数innodb_flush_log_at_tx_commit设置为1,那么在执行commit时将redo log同步写到磁盘。
bin log和redo log区别:
除了记录redo log外,当进行数据修改时还会记录undo log,undo log用于数据的撤回操作,它保留了记录修改前的内容。通过undo log可以实现事务回滚,并且可以根据undo log回溯到某个特定的版本的数据,实现MVCC。
记录所有对MySQL请求的信息,无论请求是否正确执行。
MySQL> show variables like '%general_log%';
+------------------+----------------------------------+
| Variable_name | Value |
+------------------+----------------------------------+
| general_log | OFF |
| general_log_file | /var/lib/MySQL/VM_0_7_centos.log |
+------------------+----------------------------------+
MySQL主要分为 Server 层和存储引擎层:
MySQL通过关键字将SQL语句进行解析,生成解析树。
MySQL解析器使用MySQL语法规则验证和解析查询,比如验证是否使用正确的关键字、关键字的次序是否正确和验证引号是否前后正确匹配。
预处理器会进一步检查解析树是否合法,如检查数据表和数据列是否存在,然后验证权限。
优化器会找出一个它认为最优的执行计划。
MySQL 能够处理的优化类型:
5=5 AND a > 5
转化为 a > 5
。在解析和优化阶段,MySQL将生成查询对应的执行计划,MySQL的查询执行引擎则根据这个执行计划,调用存储引擎接口来完成整个查询。
查询语句的执行流程如下:权限校验、查询缓存、分析器、优化器、权限校验、执行器、引擎。
查询语句:
select * from user where id > 1 and name = '大彬';
id > 1
还是 name = '大彬'
,优化器根据自己的优化算法选择执行效率最好的方案;更新语句执行流程如下:分析器、权限校验、执行器、引擎、redo log(prepare 状态)、binlog、redo log(commit状态)
更新语句:
update user set name = '大彬' where id = 1;
为什么记录完 redo log,不直接提交,先进入prepare状态?
假设先写 redo log 直接提交,然后写 binlog,写完 redo log 后,机器挂了,binlog 日志没有被写入,那么机器重启后,这台机器会通过 redo log 恢复数据,但是这个时候 binlog 并没有记录该数据,后续进行机器备份的时候,就会丢失这一条数据,同时主从同步也会丢失这一条数据。
假设写完了 binlog,机器异常重启了,由于没有 redo log,本机是无法恢复这一条记录的,但是 binlog 又有记录,那么和上面同样的道理,就会产生数据不一致的情况。
sql 语句查询时间超过(不包括等于) long_query_time,称为慢查询。
查看慢查询配置:
show variables like '%slow_query_log%'; #查看慢查询配置
set global slow_query_log=1; #开启慢查询
使用set global slow_query_log=1
开启了慢查询日志只对当前数据库生效,如果MySQL重启后则会失效。如果要永久生效,就必须修改配置文件my.cnf。
slow_query_log =1
slow_query_log_file=/tmp/MySQL_slow.log #系统默认会给一个缺省的文件host_name-slow.log
默认情况下long_query_time的值为10秒,可以使用命令修改,也可以在my.cnf参数里面修改。
show variables like 'long_query_time%';
set global long_query_time=4; #需要重新连接或新开一个会话才能看到修改值或者使用show global variables like 'long_query_time'
MySQL数据库支持同时两种日志存储方式,配置的时候以逗号隔开即可,如:log_output=‘FILE,TABLE’。
日志记录到系统的专用日志表中,要比记录到文件耗费更多的系统资源,因此对于需要启用慢查询日志,又需要能够获得更高的系统性能,那么建议优先记录到文件。
如果自己手动查找、分析SQL,显然是个体力活,MySQL提供了日志分析工具mysqldumpslow。
获取执行时间最长的10条sql语句:
mysqldumpslow -s al -n 10 /usr/local/MySQL/data/slow.log
分区表是一个独立的逻辑表,但是底层由多个物理子表组成。
当查询条件的数据分布在某一个分区的时候,查询引擎只会去某一个分区查询,而不是遍历整个表。在管理层面,如果需要删除某一个分区的数据,只需要删除对应的分区即可。
按照范围分区。
CREATE TABLE test_range_partition(
id INT auto_increment,
createdate DATETIME,
primary key (id,createdate)
)
PARTITION BY RANGE (TO_DAYS(createdate) ) (
PARTITION p201801 VALUES LESS THAN ( TO_DAYS('20210201') ),
PARTITION p201802 VALUES LESS THAN ( TO_DAYS('20210301') ),
PARTITION p201803 VALUES LESS THAN ( TO_DAYS('20210401') ),
PARTITION p201804 VALUES LESS THAN ( TO_DAYS('20210501') ),
PARTITION p201805 VALUES LESS THAN ( TO_DAYS('20210601') ),
PARTITION p201806 VALUES LESS THAN ( TO_DAYS('20210701') ),
PARTITION p201807 VALUES LESS THAN ( TO_DAYS('20210801') ),
PARTITION p201808 VALUES LESS THAN ( TO_DAYS('20210901') ),
PARTITION p201809 VALUES LESS THAN ( TO_DAYS('20211001') ),
PARTITION p201810 VALUES LESS THAN ( TO_DAYS('20211101') ),
PARTITION p201811 VALUES LESS THAN ( TO_DAYS('20211201') )
);
insert into test_range_partition (createdate) values ('20210105');
insert into test_range_partition (createdate) values ('20210205');
在/var/lib/mysql/data/
可以找到对应的数据文件,每个分区表都有一个使用#分隔命名的表文件:
-rw-rw---- 1 mysql mysql 65 Aug 21 09:24 db.opt
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201801.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201802.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201803.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201804.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201805.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201806.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201807.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201808.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201809.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201810.ibd
-rw-rw---- 1 mysql mysql 98304 Aug 21 09:27 test_range_partition#P#p201811.ibd
-rw-rw---- 1 mysql mysql 8598 Aug 21 09:27 test_range_partition.frm
-rw-rw---- 1 mysql mysql 116 Aug 21 09:27 test_range_partition.par
list分区。对于List分区,分区字段必须是已知的,如果插入的字段不在分区时枚举值中,将无法插入。
create table test_list_partiotion
(
id int auto_increment,
data_type tinyint,
primary key(id,data_type)
)partition by list(data_type)
(
partition p0 values in (0,1,2,3,4,5,6),
partition p1 values in (7,8,9,10,11,12),
partition p2 values in (13,14,15,16,17)
);
hash分区,可以将数据均匀地分布到预先定义的分区中。
drop table test_hash_partiotion;
create table test_hash_partiotion
(
id int auto_increment,
create_date datetime,
primary key(id,create_date)
)partition by hash(year(create_date)) partitions 10;
分区最大的优点就是优化器可以根据分区函数过滤掉一些分区,可以让查询扫描更少的数据。在查询条件中加入分区列,就可以让优化器过滤掉无需访问的分区。如果查询条件没有分区列,MySQL会让存储引擎访问这个表的所有分区。需要注意的是,查询条件中的分区列不能使用表达式。
select *
会查询出不需要的、额外的数据,那么这些额外的数据在网络上进行传输,带来了额外的网络开销。
show processlist
或 show full processlist
可以查看当前 MySQL 是否有压力,正在运行的sql,有没有慢 SQL 正在执行。
id - 线程ID,可以用:kill id;
杀死一个线程,很有用
db - 数据库
user - 用户
host - 连库的主机IP
command - 当前执行的命令,比如最常见的:Sleep,Query,Connect 等
time - 消耗时间,单位秒,很有用
state - 执行状态
sleep,线程正在等待客户端发送新的请求
query,线程正在查询或者正在将结果发送到客户端
Sorting result,线程正在对结果集进行排序
Locked,线程正在等待锁
info - 执行的SQL语句,很有用
exists 用于对外表记录做筛选。
exists 会遍历外表,将外查询表的每一行,代入内查询进行判断。当 exists 里的条件语句能够返回记录行时,条件就为真,返回外表当前记录。反之如果exists里的条件语句不能返回记录行,条件为假,则外表当前记录被丢弃。
select a.* from A a
where exists(select 1 from B b where a.id=b.id)
in 是先把后边的语句查出来放到临时表中,然后遍历临时表,将临时表的每一行,代入外查询去查找。
select * from A
where id in(select id from B)
子查询的表大的时候,使用EXISTS可以有效减少总的循环次数来提升速度;当外查询的表大的时候,使用IN可以有效减少对外查询表循环遍历来提升速度。
参考资料:
高性能MySQL书籍
MVCC实现原理:https://zhuanlan.zhihu.com/p/64576887
多版本并发控制机制:https://www.cnblogs.com/axing-articles/p/11415763.html
排他锁分析:https://blog.csdn.net/claram/article/details/54023216
分区表:https://www.cnblogs.com/wy123/p/9778590.html
一条SQL语句在MySQL中如何执行的:https://mp.weixin.qq.com/s?__biz=Mzg2OTA0Njk0OA==&mid=2247485097&idx=1&sn=84c89da477b1338bdf3e9fcd65514ac1&chksm=cea24962f9d5c074d8d3ff1ab04ee8f0d6486e3d015cfd783503685986485c11738ccb542ba7&token=79317275&lang=zh_CN#rd