隔离级别
1、事务的隔离级别决定了事务之间可见的级别。
2、当多个客户端并发地访问同一个表时,可能出现下面的一致性问题:
● 脏读取(Dirty Read)
一个事务开始读取了某行数据,但是另外一个事务已经更新了此数据但没有能够及时提交,这就出现了脏读取。
● 不可重复读(Non-repeatable Read)
在同一个事务中,同一个读操作对同一个数据的前后两次读取产生了不同的结果,这就是不可重复读。
● 幻像读(Phantom Read)
幻像读是指在同一个事务中以前没有的行,由于其他事务的提交而出现的新行。
InnoDB 实现了四个隔离级别,用以控制事务所做的修改,并将修改通告至其它并发的事务:
● 读未提交(READ UMCOMMITTED)
允许一个事务可以看到其他事务未提交的修改
● 读已提交(READ COMMITTED)
允许一个事务只能看到其他事务已经提交的修改,未提交的修改是不可见的
● 可重复读(REPEATABLE READ)
确保如果在一个事务中执行两次相同的SELECT语句,都能得到相同的结果,不管其他事务是否提交这些修改。 (银行总账)
该隔离级别为InnoDB的缺省设置
● 串行化(SERIALIZABLE) 【序列化】
将一个事务与其他事务完全地隔离。
例如:
A可以开启事物,B也可以开启事物
A在事物中执行DML语句时,未提交
B不以执行DML,DQL语句
设置服务器缺省隔离级别
● 可以在my.ini文件中使用transaction-isolation选项来设置服务器的缺省事务隔离级别
● 该选项值可以是:
READ-UNCOMMITTED
READ-COMMITTED
REPEATABLE-READ
SERIALIZABLE
● 例如:
[mysqld]
transaction-isolation = READ-COMMITTED
● 隔离级别也可以在运行的服务器中动态设置,应使用SET TRANSACTION ISOLATION LEVEL语句
● 其语法模式为:
SET [GLOBAL | SESSION] TRANSACTION ISOLATION LEVEL <isolation-level>
其中的<isolation-level>可以是:
READ UNCOMMITTED
READ COMMITTED
REPEATABLE READ
SERIALIZABLE
● 例如: SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;
1、事务隔离级别的作用范围分为两种:
● 全局级:对所有的会话有效
● 会话级:只对当前的会话有效
2、例如:设置会话级隔离级别为READ COMMITTED :
mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
或:
mysql> SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
3、设置全局级隔离级别为READ COMMITTED :
mysql> SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;
1、服务器变量tx_isolation(包括会话级和全局级两个变量)中保存着当前的会话隔离级别。
2、为了查看当前隔离级别,可访问tx_isolation变量:
● 查看会话级的当前隔离级别:
mysql> SELECT @@tx_isolation;
或:
mysql> SELECT @@session.tx_isolation;
● 查看全局级的当前隔离级别:
mysql> SELECT @@global.tx_isolation;
并发事务与隔离级别示例
read uncommitted(未提交读) --脏读(Drity Read):
会话一 |
会话二 |
mysql> prompt s1> |
mysql> use bjpowernode |
s1>use bjpowernode |
mysql> prompt s2> |
s1>create table tx ( id int(11), num int (10) ); |
|
s1>set global transaction isolation level read uncommitted; |
|
s1>start transaction; |
|
|
s2>start transaction; |
s1>insert into tx values (1,10); |
|
|
s2>select * from tx; |
s1>rollback; |
|
|
s2>select * from tx; |
read committed(已提交读)
会话一 |
会话二 |
s1> set global transaction isolation level read committed; |
|
s1>start transaction; |
|
|
s2>start transaction; |
s1>insert into tx values (1,10); |
|
s1>select * from tx; |
|
|
s2>select * from tx; |
s1>commit; |
|
|
s2>select * from tx; |
repeatable read(可重复读)
会话一 |
会话二 |
s1> set global transaction isolation level repeatable read; |
|
s1>start transaction; |
s2>start transaction; |
s1>select * from tx; |
|
s1>insert into tx values (1,10); |
|
|
s2>select * from tx; |
s1>commit; |
|
|
s2>select * from tx; |