事务

是一组原子性的SQL查询,事务里的语句,要么全部执行成功,要么全部执行失败


ACID原则:

原子性(Atomicity):事务是最小的执行单位,不允许分割。事务的原子性确保动作要么全部完成,要么完全不起作用;

一致性(Consistency):事务必须使数据库从一个一致性状态变换到另一个一致性状态,也就是说一个事务执行之前和执行之后都必须处于一致性状态。就像小明余额减去200与支出增加200,相加与之前是一样的。

隔离性(Isolation):并发访问数据库时,一个用户的事务不被其他事务所干扰,各并发事务之间数据库是独立的;

持久性(Durability):一个事务被提交之后。它对数据库中数据的改变是持久的,即使数据库发生故障也不应该对其有任何影响。


冲突问题

脏读 :

一个事务读取到另外一个事务未提交的数据

解决:就是把释放锁的位置调整到事务提交之后,此时在事务提交前,其他进程是无法对该行数据进行读取的,包括任何操作


不可重复读:

一个事务读取到另外一个事务已经提交的数据,也就是说一个事务可以看到其他事务所做的修改

在一个事务内,多次读同一个数据。

在这个事务还没有结束时,另一个事务也访问该同一数据。那么,在第一个事务的两次读数据之间。由于第二个事务的修改,那么第一个事务读到的数据可能不一样,这样就发生了在一个事务内两次读到的数据是不一样的,因此称为不可重复读,即原始读取不可重复。


虚读(幻读):

是指在一个事务内读取到了别的事务插入的数据,导致前后读取不一致。

和不可重复读类似,但虚读(幻读)会读到其他事务的插入的数据,导致前后读取不一致

如第一个事务对一个表中的数据进行了修改,比如这种修改涉及到表中的“全部数据行”。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入“一行 新数据”。那么,以后就会发生操作第一个事务的用户发现表中还存在没有修改的数据行,就好象发生了幻觉一样.一般解决幻读的方法是增加范围锁 RangeS,锁定检索范围为只读,这样就避免了幻读。


事务隔离级别有4种:

Read uncommitted(读未提交): 会出现脏读,不可重复读,幻读

Read committed(读已提交):会出现不可重复读,幻读

Repeatable read(可重复读):会出现幻读(但在Mysql实现的Repeatable read配合间隙锁不会出现幻读!)

Serializable(串行):避免以上的情况!

设置事务隔离级别: set  [glogal | session]  transaction isolation level  (Read uncommitted);

死锁:两个或多个以上的事务在同一资源上相互占用,并请求锁定对方占用的资源,从而导致恶性循环。

解决:将持有最少行级排他锁的事务进行回滚。

事务日志:可以提高事务的效率。在修改表时只需要修改其内存拷贝,再把该修改行为记录到持久在硬盘上的事务日志中。事务日志被持久以后,内存中被修改的数据在后台可以慢慢的刷回磁盘。



什么是MVCC?

MVCC,全称Multi-Version Concurrency Control,即多版本并发控制。

MVCC是一种并发控制的方法,一般在数据库管理系统中,实现对数据库的并发访问,在编程语言中实现事务内存。 

MVCC在MySQL InnoDB中的实现主要是为了提高数据库并发性能,用更好的方式去处理读-写冲突,做到即使有读写冲突时,也能做到不加锁,非阻塞并发读


什么是当前读和快照读?

在学习MVCC多版本并发控制之前,我们必须先了解一下,什么是MySQL InnoDB下的当前读和快照读?

当前读

像select lock in share mode(共享锁), select for update ; update, insert ,delete(排他锁)这些操作都是一种当前读,为什么叫当前读?

就是它读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。


快照读

像不加锁的select操作就是快照读,即不加锁的非阻塞读;

快照读的前提是隔离级别不是串行级别,串行级别下的快照读会退化成当前读;

之所以出现快照读的情况,是基于提高并发性能的考虑,快照读的实现是基于多版本并发控制,即MVCC,可以认为MVCC是行锁的一个变种,但它在很多情况下,避免了加锁操作,降低了开销;

既然是基于多版本,即快照读可能读到的并不一定是数据的最新版本,而有可能是之前的历史版本

说白了MVCC就是为了实现读-写冲突不加锁,而这个读指的就是快照读, 而非当前读,当前读实际上是一种加锁的操作,是悲观锁的实现


当前读,快照读和MVCC的关系

  • 准确的说,MVCC多版本并发控制指的是 “维持一个数据的多个版本,使得读写操作没有冲突” 这么一个概念。仅仅是一个理想概念
  • 而在MySQL中,实现这么一个MVCC理想概念,我们就需要MySQL提供具体的功能去实现它,而快照读就是MySQL为我们实现MVCC理想模型的其中一个具体非阻塞读功能。而相对而言,当前读就是悲观锁的具体功能实现


MVCC能解决什么问题,好处是?

数据库并发场景有三种,分别为:

  • 读-读:不存在任何问题,也不需要并发控制
  • 读-写:有线程安全问题,可能会造成事务隔离性问题,可能遇到脏读,幻读,不可重复读
  • 写-写:有线程安全问题,可能会存在更新丢失问题,比如第一类更新丢失,第二类更新丢失


MVCC带来的好处是?

多版本并发控制(MVCC)是一种用来解决读-写冲突的无锁并发控制,也就是为事务分配单向增长的时间戳,为每个修改保存一个版本,版本与事务时间戳关联,读操作只读该事务开始前的数据库的快照。 所以MVCC可以为数据库解决以下问题

  • 在并发读写数据库时,可以做到在读操作时不用阻塞写操作,写操作也不用阻塞读操作,提高了数据库并发读写的性能
  • 同时还可以解决脏读,幻读,不可重复读等事务隔离问题,但不能解决更新丢失问题


小结一下咯

总之,MVCC就是因为大牛们,不满意只让数据库采用悲观锁这样性能不佳的形式去解决读-写冲突问题,而提出的解决方案,所以在数据库中,因为有了MVCC,所以我们可以形成两个组合:

MVCC + 悲观锁

MVCC解决读写冲突,悲观锁解决写写冲突


MVCC + 乐观锁

MVCC解决读写冲突,乐观锁解决写写冲突

这种组合的方式就可以最大程度的提高数据库并发性能,并解决读写冲突,和写写冲突导致的问题



​https://www.jianshu.com/p/8845ddca3b23​


MVCC多版本并发控制指的是:维持一个数据的多个版本,使得读写操作没有“冲突”,是一个理想概念;


MVCC是一种用来解决读-写冲突的无锁并发控制,读操作只读该事物开始前的数据库的快照。

在并发读写数据库时,可以做到在读操作时不用阻塞写操作,写操作也不用阻塞读操作,提高了数据库并发读写的性能;

同时还可以解决脏读、幻读、不可重复读事务隔离问题,

但不能解决更新丢失问题;

​https://mp.weixin.qq.com/s/A_RjFxZ45iEms-irRgzZDQ​