- update undo log
事务在进行update或delete时产生的undo log; 不仅在事务回滚时需要,在快照读时也需要;所以不能随便删除,只有在快速读或事务回滚不涉及该日志时,对应的日志才会被purge线程统一清除
三、MySQL中的索引
索引的常见模型有哈希表
、有序数组
和搜索树
。
哈希表
:一种以KV存储数据的结构,只适合等值查询,不适合范围查询。
有序数组
:只适用于静态存储引擎,涉及到插入的时候比较麻烦。可以参考Java中的ArrayList。
搜索树
:按照数据结构中的二叉树来存储数据,不过此时是N叉树(B+树)。广泛应用在存储引擎层中。
B+树比B树优势
在于:
- B+ 树非叶子节点存储的只是索引,可以存储的更多。B+树比B树更加矮胖,IO次数更少。
- B+ 树叶子节点前后管理,更加方便范围查询。同时结果都在叶子节点,查询效率稳定。
- B+树中更有利于对数据扫描,可以避免B树的回溯扫描。
索引的优点:
1、唯一索引可以保证每一行数据的唯一性
2、提高查询速度
3、加速表与表的连接
4、显著的减少查询中分组和排序的时间
5、通过使用索引,可以在查询的过程中,使用优化隐藏器,提高系统的性能。
索引的缺点:
1、创建跟维护都需要耗时
2、创建索引时,需要对表加锁,在锁表的同时,可能会影响到其他的数据操作
3、 索引需要磁盘的空间进行存储,磁盘占用也很快。
4、当对表中的数据进行CRUD的时,也会触发索引的维护,而维护索引需要时间,可能会降低数据操作性能
索引设计的原则不应该:
1、索引不是越多越好。索引太多,维护索引需要时间跟空间。
2、 频繁更新的数据,不宜建索引。
3、数据量小的表没必要建立索引。
应该:
1、重复率小的列建议生成索引。因为重复数据少,索引树查询更有效率,等价基数越大越好。
2、数据具有唯一性,建议生成唯一性索引。在数据库的层面,保证数据正确性
3、频繁group by、order by的列建议生成索引。可以大幅提高分组和排序效率
4、经常用于查询条件的字段建议生成索引。通过索引查询,速度更快
索引失效的场景
1、
模糊搜索
:左模糊或全模糊都会导致索引失效,比如’%a’和’%a%‘。但是右模糊是可以利用索引的,比如’a%’ 。2、
隐式类型转换
:比如select * from t where name = xxx , name是字符串类型,但是没有加引号,所以是由MySQL隐式转换的,所以会让索引失效 3、当语句中带有or的时候
:比如select * from t where name=‘sw’ or age=144、
不符合联合索引的最左前缀匹配
:(A,B,C)的联合索引,你只where了C或B或只有B,C
关于索引的知识点:
主键索引
:主键索引的叶子节点存的是整行
数据信息。在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不支持聚簇索引
四、SQL事务隔离级别
ACID的四个特性
原子性
(Atomicity):把多个操作放到一个事务中,保证这些操作要么都成功,要么都不成功一致性
(Consistency):理解成一串对数据进行操作的程序执行下来,不会对数据产生不好的影响,比如凭空产生,或消失隔离性
(Isolation,又称独立性):隔离性的意思就是多个事务之间互相不干扰,即使是并发事务的情况下,他们只是两个并发执行没有交集,互不影响的东西;当然实现中,也不一定需要这么完整隔离性,即不一定需要这么的互不干扰,有时候还是允许有部分干扰的。所以MySQL可以支持4种事务隔离性持久性
(Durability):当某个操作操作完毕了,那么结果就是这样了,并且这个操作会持久化到日志记录中
PS:ACID中C与CAP定理中C的区别
ACID的C着重强调单数据库事务操作时,要保证数据的完整和正确性,数据不会凭空消失跟增加。CAP
理论中的C指的是对一个数据多个备份的读写一致性
事务操作可能会出现的数据问题
1、
脏读
(dirty read):B事务更改数据还未提交,A事务已经看到并且用了。B事务如果回滚,则A事务做错了2、
不可重复读
(non-repeatable read):不可重复读的重点是修改: 同样的条件, 你读取过的数据, 再次读取出来发现值不一样了,只需要锁住满足条件的记录3、
幻读
(phantom read):事务A先修改了某个表的所有纪录的状态字段为已处理,未提交;事务B也在此时新增了一条未处理的记录,并提交了;事务A随后查询记录,却发现有一条记录是未处理的造成幻读现象,幻读仅专指新插入的行
。幻读会造成语义上
的问题跟数据一致性
问题。4、 在可重复读RR隔离级别下,普通查询是
快照读
,是不会看到别的事务插入的数据的。因此,幻读在当前读
下才会出现。要用间隙锁解决此问题。
在说隔离级别之前,你首先要知道,你隔离得越严实,效率就会越低
。因此很多时候,我们都要在二者之间寻找一个平衡点。SQL标准的事务隔离级别由低到高如下:
上图从上到下的模式会导致系统的并行性能依次降低,安全性依次提高。
读未提交
:别人改数据的事务尚未提交,我在我的事务中也能读到。
读已提交(Oracle默认)
:别人改数据的事务已经提交,我在我的事务中才能读到。
可重复读(MySQL默认)
:别人改数据的事务已经提交,我在我的事务中也不去读,以此保证重复读一致性。
串行
:我的事务尚未提交,别人就别想改数据。
标准跟实现
:上面都是关于事务的标准,但是每一种数据库都有不同的实现,比如MySQL InnDB
默认为RR
级别,但是不会出现幻读。因为当事务A更新了所有记录的某个字段,此时事务A会获得对这个表的表锁,因为事务A还没有提交,所以事务A获得的锁没有释放,此时事务B在该表插入新记录,会因为无法获得该表的锁,则导致插入操作被阻塞。只有事务A提交了事务后,释放了锁,事务B才能进行接下去的操作。所以可以说 MySQL的RR级别的隔离是已经实现解决了脏读,不可重复读和幻读的。
五、MySQL中的锁
无论是Java的并发编程还是数据库的并发操作都会涉及到锁,研发人员引入了悲观锁
跟乐观锁
这样一种锁的设计思想。
悲观锁:
优点
:适合在写多读少的并发环境中使用,虽然无法维持非常高的性能,但是在乐观锁无法提更好的性能前提下,可以做到数据的安全性
缺点
:加锁会增加系统开销,虽然能保证数据的安全,但数据处理吞吐量低,不适合在读书写少的场合下使用
乐观锁:
优点
:在读多写少的并发场景下,可以避免数据库加锁的开销,提高DAO层的响应性能,很多情况下ORM工具都有带有乐观锁的实现,所以这些方法不一定需要我们人为的去实现。
缺点
:在写多读少的并发场景下,即在写操作竞争激烈的情况下,会导致CAS多次重试,冲突频率过高,导致开销比悲观锁更高。
实现
:数据库层面的乐观锁其实跟CAS
思想类似, 通数据版本号
或者时间戳
也可以实现。
数据库并发场景主要有三种:
读-读
:不存在任何问题,也不需要并发控制
读-写
:有隔离性问题,可能遇到脏读,幻读,不可重复读
写-写
:可能存更新丢失问题,比如第一类更新丢失,第二类更新丢失
两类更新丢失问题:
第一类更新丢失:事务A的事务回滚覆盖了事务B已提交的结果 第二类更新丢失:事务A的提交覆盖了事务B已提交的结果
为了合理贯彻落实锁的思想,MySQL中引入了杂七杂八的各种锁:
锁分类
MySQL支持三种层级的锁定,分别为
1.表级锁定
MySQL中锁定粒度
最大
的一种锁,最常使用的MYISAM与INNODB都支持表级锁定。
2.页级锁定
是MySQL中锁定粒度介于行级锁和表级锁中间的一种锁,表级锁速度快,但冲突多,行级冲突少,但速度慢。所以取了
折衷
的页级,一次锁定相邻的一组记录。
3.行级锁定
Mysql中锁定粒度
最细
的一种锁,表示只针对当前操作的行进行加锁。行级锁能大大减少数据库操作的冲突。其加锁粒度最小,但加锁的开销也最大行级锁不一定比表级锁要好
:锁的粒度越细,代价越高,相比表级锁在表的头部直接加锁,行级锁还要扫描找到对应的行对其上锁,这样的代价其实是比较高的,所以表锁和行锁各有所长。
MyISAM中的锁
-
虽然MySQL支持表,页,行三级锁定,但MyISAM存储引擎只支持表锁。所以MyISAM的加锁相对比较开销低,但数据操作的并发性能相对就不高。但如果写操作都是尾插入,那还是可以支持一定程度的读写并发
-
从MyISAM所支持的锁中也可以看出,MyISAM是一个支持读读并发,但不支持通用读写并发,写写并发的数据库引擎,所以它更适合用于读多写少的应用场合,一般工程中也用的较少。
InnoDB中的锁
该模式下支持的锁实在是太多了,具体如下:
共享锁和排他锁 (Shared and Exclusive Locks)
意向锁(Intention Locks)
记录锁(Record Locks)
间隙锁(Gap Locks)
临键锁 (Next-Key Locks)
插入意向锁(Insert Intention Locks)
主键自增锁 (AUTO-INC Locks)
空间索引断言锁(Predicate Locks for Spatial Indexes)
举个栗子,比如行锁里的共享锁跟排它锁:lock in share modle
共享读锁:
为了确保自己查到的数据没有被其他的事务正在修改,也就是说确保查到的数据是
最新的数据
,并且不允许其他人来修改数据。但是自己不一定能够修改数据,因为有可能其他的事务也对这些数据使用了in share mode
的方式上了S
锁。如果不及时的commit 或者rollback 也可能会造成大量的事务等待。
for update
排它写锁:
为了让自己查到的数据确保是最新数据,并且查到后的数据只允许自己来修改的时候,需要用到
for update
。相当于一个 update 语句。在业务繁忙的情况下,如果事务没有及时的commit或者rollback 可能会造成其他事务长时间的等待,从而影响数据库的并发使用效率。
Gap Lock
间隙锁:
1、行锁只能锁住行,如果在记录之间的间隙插入数据就无法解决了,因此MySQL引入了间隙锁(Gap Lock)。间隙锁是
左右开区间
。间隙锁之间不会冲突
。2、间隙锁和行锁合称
NextKeyLock
,每个NextKeyLock
是前开后闭区间
。
间隙锁加锁原则(学完忘那种):
1、加锁的基本单位是 NextKeyLock,是前开后闭区间。
2、查找过程中访问到的对象才会加锁。
3、索引上的等值查询,给
唯一索引
加锁的时候,NextKeyLock退化为行锁。4、索引上的等值查询,向右遍历时且最后一个值不满足等值条件的时候,NextKeyLock退化为间隙锁。
5、唯一索引上的范围查询会访问到不满足条件的第一个值为止。
六、MVCC
MVCC:
1、全称
Multi-Version Concurrency Control
,即多版本并发控制
。MVCC是一种并发控制的理念
,维持一个数据的多个版本,使得读写操作没有冲突。2、MVCC在MySQL InnoDB中实现目的主要是为了提高数据库并发性能,用更好的方式去处理读-写冲突,做到即使有读写冲突时,也能做到不加锁,非阻塞并发读。
MySQL InnoDB下的当前读和快照读
1.当前读
1、像
select lock in share mode
(共享锁)、select for updat
e 、update
、insert
、delete
(排他锁)这些操作都是一种当前读
,就是它读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁
。2、当前读可以认为是
悲观锁
的具体功能实现
2.快照读
1、不加锁的select就是快照读,即不加锁的非阻塞读;快照读的前提是隔离级别不是串行级别,串行级别下的快照读会退化成当前读;之所以出现快照读的情况,是基于提高并发性能的考虑,快照读的实现是基于多版本并发控制,即
MVCC
,可以认为MVCC是行锁的一个变种
,但它在很多情况下,避免了加锁操作
,降低了开销;既然是基于多版本,即快照读可能读到的并不一定是数据的最新版本,而有可能是之前的历史版本。2、快照读就是MVCC思想在MySQL的具体非阻塞读功能实现,MVCC的目的就是为了实现读-写冲突不加锁,提高并发读写性能,而这个读指的就是
快照读
。3、快照读就是MySQL为我们实现MVCC理想模型的其中一个具体非阻塞读功能。
因为大佬不满意只让数据库采用悲观锁这样性能不佳的形式去解决读-写冲突问题,而提出了MVCC,所以我们可以形成两个组合:
MVCC + 悲观锁
:MVCC解决读写冲突,悲观锁解决写写冲突
MVCC + 乐观锁
:MVCC解决读写冲突,乐观锁解决写写冲突
MVCC的实现原理
MVCC实现原理主要是依赖记录中的 四个隐式字段
、undo日志
、Consistent Read View
来实现的。
四个隐式字段:
1.DB_TRX_ID:
6byte,最近修改(修改/插入)事务ID:记录创建这条记录/最后一次修改该记录的
事务ID
2.DB_ROLL_PTR
7byte,回滚指针,指向这条记录的
上一个版本
(存储于rollback segment里)
3.DB_ROW_ID
6byte,隐含的自增ID(
隐藏主键
),如果数据表没有主键,InnoDB会自动以DB_ROW_ID产生一个聚簇索引
4.FLAG
一个删除flag隐藏字段, 既记录被更新或删除并不代表真的删除,而是删除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
。
1.insert时:
InnoDB为新插入的每一行保存当前系统版本号作为版本号。
2.select时:
1、 InnoDB只会查找版本早于当前事务版本的数据行(也就是行的系统版本号
<=
事务的系统版本号),这样可以确保事务读取的行,要么是在事务开始前已经存在的,要么是事务自身插入或者修改过的。2、行的删除版本要么未定义,要么大于当前事务版本号,这可以确保事务读取到的行在事务开始之前未被删除。
3、只有1,2 同时满足的记录,才能返回作为查询结果。
3.delete时:
InnoDB会为删除的每一行保存当前系统的版本号(事务的ID)作为删除标识.
4.update时:
InnoDB执行update,实际上是新插入了一行记录,并保存其创建时间为当前事务的ID,同时保存当前事务ID到要update的行的删除时间。
上面只是一个浅显的讲解MVCC选择标准流程,源码层面应该是根据低水位
跟高水位
来截取的。具体实现可自行百度。
重点
:
1、事务中快照读的结果是
非常依赖
该事务首次出现快照读的地方,即某个事务中首次出现快照读的地方非常关键,它有决定该事务后续快照读结果的能力。2、在
RC
隔离级别下,是每个快照读都会生成
并获取最新的Read View;而在RR
隔离级别下,则是同一个事务中的第一个
快照读才会创建Read View, 之后的快照读获取的都是同一个
Read View。
七、缓冲池(buffer pool)
应用系统分层架构,为了加速数据访问,会把最常访问的数据,放在缓存(cache)里,避免每次都去访问数据库。操作系统,会有缓冲池(buffer pool)机制,避免每次访问磁盘,以加速数据的访问。MySQL作为一个存储系统,同样具有缓冲池(buffer pool)机制,以避免每次查询数据都进行磁盘IO,主要作用:
1、存在的意义是加速查询
2、缓冲池(buffer pool) 是一种常见的降低磁盘访问 的机制;
3、缓冲池通常以页(page 16K)为单位缓存数据;
4、缓冲池的常见管理算法是LRU,memcache,OS,InnoDB都使用了这种算法;
5、InnoDB对普通LRU进行了优化:将缓冲池分为
老生代
和新生代
,入缓冲池的页,优先进入老生代,该页被访问,才进入新生代,以解决预读失效的问题页被访问。且在老生代停留时间超过配置阈值的,才进入新生代,以解决批量数据访问,大量热数据淘汰的问题
预读失效:
由于预读(Read-Ahead),提前把页放入了缓冲池,但最终MySQL并没有从页中读取数据,称为预读失效
缓冲池污染:
当某一个SQL语句,要批量扫描大量数据时,可能导致把缓冲池的所有页都替换出去,导致大量热数据被换出,MySQL性能急剧下降,这种情况叫缓冲池污染。解决办法:加入
老生代停留时间窗口
策略后,短时间内被大量加载的页,并不会立刻插入新生代头部,而是优先淘汰那些,短期内仅仅访问了一次的页。
八、table瘦身
空洞:
MySQL执行
delete
命令其实只是把记录的位置,或者数据页标记为了可复用
,但磁盘文件的大小是不会变的。通过delete命令是不能回收表空间的。这些可以复用,而没有被使用的空间,看起来就像是空洞
。插入时候引发分裂同样会产生空洞。
重建表思路:
1、新建一个跟A表结构相同的表B
2、按照主键ID将A数据一行行读取同步到表B
3、用表B替换表A实现效果上的瘦身。
重建表指令:
1、alter table A engine=InnoDB,慎重用,牛逼的DBA都用下面的开源工具。
2、推荐Github:gh-ost
九、SQL Joins、统计、 随机查询
7种join具体如下:
统计:
1、MyISAM模式下把一个表的总行数存在了磁盘上,直接拿来用即可
2、InnoDB引擎由于 MVCC的原因,需要把数据读出来然后累计求和
3、性能来说 由坏到好:count(字段) < count(主键id) < count(1) ≈ count(*),
尽量用count(*)即可。
随机查询:
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());
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
总结
其他的内容都可以按照路线图里面整理出来的知识点逐一去熟悉,学习,消化,不建议你去看书学习,最好是多看一些视频,把不懂地方反复看,学习了一节视频内容第二天一定要去复习,并总结成思维导图,形成树状知识网络结构,方便日后复习。
这里还有一份很不错的《Java基础核心总结笔记》,特意跟大家分享出来
目录:
部分内容截图:
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
的内容都可以按照路线图里面整理出来的知识点逐一去熟悉,学习,消化,不建议你去看书学习,最好是多看一些视频,把不懂地方反复看,学习了一节视频内容第二天一定要去复习,并总结成思维导图,形成树状知识网络结构,方便日后复习。
这里还有一份很不错的《Java基础核心总结笔记》,特意跟大家分享出来
目录:
[外链图片转存中…(img-sSmVmJG6-1712997239448)]
部分内容截图:
[外链图片转存中…(img-r66DU5sh-1712997239448)]
[外链图片转存中…(img-ybtKURgq-1712997239449)]
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-eSbudis3-1712997239449)]