《MySQL实战45讲》读后感 03|事务隔离:为什么你改了我还看不见

收获到的知识点

  • 事务支持是在引擎层面实现的,但是并不是所有的引擎都支持事务,比如MySQL原生的MyISAM引擎就不支持事务,这也是MyISAM被 InnoDB取代的原因之一
  • 多个事务同时执行的时候,可能出现这3种情况:脏读(dirty read)、不可重复读(non-repeatable read)、幻读(phanton read),为了解决这些总是就有了“隔离级别”的概念
  • 隔离级别有哪些级别:读未提交(read uncommitted)、读已提交(read committed)、可重复读(repeatable read)、可串行化(serializable)

隔离级别产生的效果

我们来看看在不同的隔离级别下,事务 A 会有哪些不同的返回结果,也就是图里面 V1、V2、V3 的返回值分别是什么。
在这里插入图片描述
若隔离级别是“读未提交”, 则 V1 的值就是 2。这时候事务 B 虽然还没有提交,但是结果已经被 A 看到了。因此,V2、V3 也都是 2。
若隔离级别是“读提交”,则 V1 是 1,V2 的值是 2。事务 B 的更新在提交后才能被 A 看到。所以, V3 的值也是 2。
若隔离级别是“可重复读”,则 V1、V2 是 1,V3 是 2。之所以 V2 还是 1,遵循的就是这个要求:事务在执行期间看到的数据前后必须是一致的。
若隔离级别是“串行化”,则在事务 B 执行“将 1 改成 2”的时候,会被锁住。直到事务 A 提交后,事务 B 才可以继续执行。所以从 A 的角度看, V1、V2 值是 1,V3 的值是 2。

隔离级别的实战练习:

准备工作:

在这里插入图片描述

隔离级别不同那么如何实现的:
在实现上,数据库里面会创建一个视图,访问的时候以视图的逻辑结果为准。在“可重复读”隔离级别下,这个视图是在事务启动时创建的,整个事务存在期间都用这个视图。在“读提交”隔离级别下,这个视图是在每个 SQL 语句开始执行的时候创建的。这里需要注意的是,“读未提交”隔离级别下直接返回记录上的最新值,没有视图概念;而“串行化”隔离级别下直接用加锁的方式来避免并行访问。

事务的启动方式:

MySQL 的事务启动方式有以下几种:
显式启动事务语句, begin 或 start transaction。配套的提交语句是 commit,回滚语句是 rollback。
set autocommit=0,这个命令会将这个线程的自动提交关掉。意味着如果你只执行一个 select 语句,这个事务就启动了,而且并不会自动提交。这个事务持续存在直到你主动执行 commit 或 rollback 语句,或者断开连接。

有些客户端连接框架会默认连接成功后先执行一个 set autocommit=0 的命令。这就导致接下来的查询都在事务中,如果是长连接,就导致了意外的长事务。
因此,我会建议你总是使用 set autocommit=1, 通过显式语句的方式来启动事务。

课堂问题

Q:你现在知道了系统里面应该避免长事务,如果你是业务开发负责人同时也是数据库负责人,你会有什么方案来避免出现或者处理这种情况呢?
A:事前写代码环节,可以不用事务的地方就不要加事务,比如一些独立的select操作,合理使用数据库连接池dbpool(c3p0,dbcp,druid),在需要启用事物的地方才开启事物,try{setautocommit(false)}finly{setautocommit(true)};事后增加db监控,定期就监控到的长事物进行review验证其合理性。

顺便分享一下林奇大师的课程,有兴趣的可以看看

在这里插入图片描述

发布了100 篇原创文章 · 获赞 64 · 访问量 25万+

猜你喜欢

转载自blog.csdn.net/hl_java/article/details/84556557
今日推荐