推荐(免费):mysql视频教程
事务的隔离性
MySQL是一个客户端/服务器架构的软件,对于同一个服务器来说,可以有若干个客户端与之连接,每个客户端与服务器连接上之后,就可以称之为一个会话(Session)。每本文来源[email protected]搞@^&代*@码2网个客户端都可以在自己的会话中向服务器发出请求语句,一个请求语句可能是某个事务的一部分,也就是对于服务器来说可能同时处理多个事务。当数据库上有多个事务同时执行的时候,就可能出现脏读(Dirty Read)、不可重复读(Non-Repeatable Read)、幻读(Phantom Read)的问题,为了解决这些问题,就有了 “隔离级别” 的概念。
理论上在某个事务对某个数据进行访问时,其他事务应该进行排队,当该事务提交之后,其他事务才可以继续访问这个数据。但一般情况下隔离得越严实,效率就会越低。因此很多时候,我们都要在隔离性和效率二者之间寻找一个平衡点。
事务并发执行遇到的问题
脏读(Dirty Read): 脏读是指一个事务读到了另一个未提交事务修改过的数据。
如小王的账户中有100的余额,接下来有两个事务对小王的账户进行访问。
会话A | 会话B |
---|---|
begin; | |
update xxx set balance = balance+50 where client_no = ‘小王客户号’ ; | begin; |
select balance from xxx where client_no = ‘小王客户号’ ; (如果读到150,则意味着发生了脏读) | |
rollback; | commit; |
如上,会话A和会话B各开启了一个事务,会话A先给小王账户余额加了50,此时账户B查询小王账户余额为150,接下来会话A进行了回滚,那会话B查询到的150就成一个不正确的脏数据。
不可重复读(Non-Repeatable Read): 不可重复读是指在同一个事务内多次读取同一数据集合,但查到的结果却不相同。发生不可重复读的原因是在多次搜索期间查询的数据被其它事务修改了。
看如下的两个会话请求。
会话A | 会话B |
---|---|
begin; | |
select balance from xxx where client_no = ‘小王客户号’ ; (读到余额为100) | begin; |
update xxx set balance = balance+50 where client_no = ‘小王客户号’ ; | |
commit; | |
select balance from xxx where client_no = ‘小王客户号’ ; (如果读到150,则意味着发生了不可重复读) | |
commit; |
在会话A的同一个事务中,两次相同查询的结果不同,意味着发生了不可重复读。
幻读(Phantom Read): 所谓幻读,指的是当某个事务在读取某个范围内的记录时,另外一个事务又在该范围内插入了新的记录,当之前的事务再次读取该范围的记录时,会读取到之前没有读到的数据。
假如账户表中目前只有小王的余额为100,再看下如下的两个会话请求。
会话A | 会话B |
---|---|
begin; | |
select name from xxx where balance = 100 ; (读到name为‘小王’) | begin; |
insert into xxx(client_no,name,balance) values(‘小张客户号’,‘小张’,100); | |
commit; | |
select name from xxx where balance = 100 ; (如果读到了‘小王’和‘小张’,则意味着发生了幻读) | |
commit; |