MySQL InnoDB事务的隔离级别

  MySQL InnoDB事务的隔离级别有四级,默认是“可重复读”(REPEATABLE READ)。

  • 未提交读(READ UNCOMMITTED)。另一个事务修改了数据,但尚未提交,而本事务中的SELECT会读到这些未被提交的数据(脏读)。
  • 提交读(READ COMMITTED)。本事务读取到的是最新的数据(其他事务提交后的)。问题是,在同一个事务里,前后两次相同的SELECT会读到不同的结果(不重复读)。如两个会话a和b,由于是read committed所以只有当事务提交后才能被别的事务可见,当a执行查询后b执行插入,b执行commit提交事务,这时a再次查询结果确实不一样的,a的两次查询同属于一个事务,即为不可重复读。
  • 可重复读(REPEATABLE READ)。在同一个事务里,SELECT的结果是事务开始时时间点的状态,因此,同样的SELECT操作读到的结果会是一致的。直到提交事务后再查询才可以看到其他事务在本事务执行期间锁进行的更改操作。在MySQL中InnoDB和XtraDB存储引擎通过多版本并发控制(MVCC)解决了幻读的问题。
  • 串行化(SERIALIZABLE)。读操作会隐式获取共享锁,可以保证不同事务间的互斥。
  • SQL 事务隔离级别 

     

    离级别 脏读(Dirty Read)         不可重复读(NonRepeatable Read)幻读(Phantom Read)
    读未提交(Read uncommitted)可能可能可能
    读已提交(Read committed)不可能可能可能
    可重复读(Repeatable read)不可能不可能可能
    可串行化(Serializable )不可能不可能不可能

     

    在MySQL中默认事务隔离级别是可重复读(Repeatable read).可通过SQL语句查询:

    查看InnoDB系统级别的事务隔离级别:

     

        mysql> SELECT @@global.tx_isolation;

     

    结果:

    +-----------------------+
    | @@global.tx_isolation |
    +-----------------------+
    | REPEATABLE-READ       |
    +-----------------------+
    1 row in set (0.00 sec)

     

    查看InnoDB会话级别的事务隔离级别:

     

      mysql> SELECT @@tx_isolation;

     

    结果:
    +-----------------+
    | @@tx_isolation  |
    +-----------------+
    | REPEATABLE-READ |
    +-----------------+
    1 row in set (0.00 sec)

     

    修改事务隔离级别:


        mysql> set global transaction isolation level read committed;


        Query OK, 0 rows affected (0.00 sec)
     

        mysql> set session transaction isolation level read committed;
     

        Query OK, 0 rows affected (0.00 sec)

     

    InnoDB 的可重复读隔离级别和其他数据库的可重复读是有区别的,不会造成幻象读(phantom read),所谓幻象读,就是同一个事务内,多次select,可以读取到其他session insert并已经commit的数据。下面是一个小的测试,证明InnoDB的可重复读隔离级别不会造成幻象读。测试涉及两个session,分别为 session 1和session 2,隔离级别都是repeateable read,关闭autocommit

     

        mysql> select @@tx_isolation;
      

        +-----------------+
        | @@tx_isolation  |
        +-----------------+
        | REPEATABLE-READ |
        +-----------------+
        1 row in set (0.00 sec)
       

        mysql> set autocommit=off;


        Query OK, 0 rows affected (0.00 sec)

     

    session 1 创建表并插入测试数据


         mysql> create table test(i int) engine=innodb;

        Query OK, 0 rows affected (0.00 sec)


        mysql> insert into test values(1);
        Query OK, 1 row affected (0.00 sec)

     

    session 2 查询,没有数据,正常,session1没有提交,不允许脏读
         mysql> select * from test;
         Empty set (0.00 sec)

     

    session 1 提交事务


        mysql> commit;
        Query OK, 0 rows affected (0.00 sec)

     

    session 2 查询,还是没有数据,没有产生幻象读
                            
        mysql> select * from test;
        Empty set (0.00 sec)
                            
    当session2提交事务后才可以看到session1的插入数据;                       
    以上试验版本:
                            
        mysql> select version();
        +-------------------------+
        | version()               |
        +-------------------------+
        | 5.0.37-community-nt-log |
        +-------------------------+
        1 row in set (0.00 sec)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值