zoukankan      html  css  js  c++  java
  • 四种事务的隔离级别

    SQL 标准定义的四个隔离级别为: Read Uncommitted (读未提交 ) 、 Read Committed (读已提交)、 Repeatable Read (可重复读)、 Serializable (可串行化) ,下面分别介绍。

    准备: 创建一张用于测试的查询表 account表,字段自定义,并维护简单的初始化数据以供查询使用。

    1、未提交:

    指一个事务可以读取到另一个未提交事务的数据。

    • 打开一个客户端A,并设置当前事务模式为read uncommitted(未提交读),查询表account的初始值:
    • 在客户端A的事务提交之前,打开另一个客户端B,更新表account:
    • 这时,虽然客户端B的事务还没提交,但是客户端A就可以查询到B已经更新的数据。
    • 一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那客户端A查询到的数据其实就是脏读数据。  

      

    2、读提交

    指一个事务会读取到其他事务已提交的数据,但对于其他事务中未提交的不会再读到。

    • 打开一个客户端A,并设置当前事务模式为read committed(未提交读),查询表account的所有记录;
    • 在客户端A的事务提交之前,打开另一个客户端B,更新表account;
    • 这时,客户端B的事务还没提交,客户端A不能查询到B已经更新的数据,解决了脏读问题;
    • 客户端B的事务提交;
    • 客户端A执行与上一步相同的查询,结果 与上一步不一致,即产生了不可重复读的问题。

    3、重复读

    指在同一事务中,相同的读取操作前后结果保持一致。

    • 打开一个客户端A,并设置当前事务模式为repeatable read,查询表account的所有记录
    • 在客户端A的事务提交之前,打开另一个客户端B,更新表account并提交。
    • 在客户端A查询表account的所有记录,与步骤(1)查询结果一致,没有出现不可重复读的问题。
    • l 可重复读会对操作行进行加锁,对于被操作的这条行记录是不允许修改的。写到这里,应该明白的一点就是,不可重复读对应的是修改,即UPDATE操作。但是可能还会有幻读问题。因为幻读问题对应的是插入INSERT操作,而不是UPDATE操作。(这里的幻读指的是理论上存在,对于mysql等成熟的数据库,已经通过更优的算法避免了幻读)
    • 重新打开客户端B,插入一条新数据后提交;
    • 在客户端A查询表account的所有记录,没有 查出 新增数据,所以没有出现幻读

    4、Serializable 序列化(串行化)

    序列化是最高的事务隔离级别,在该级别下,事务串行化顺序执行,可以避免脏读、不可重复读与幻读。但是这种事务隔离级别效率低下,比较耗数据库性能,一般不使用。

    总结:

    1、事务隔离级别为读提交时,写数据只会锁住相应的行;

    2、事务隔离级别为可重复读时,如果检索条件有索引(包括主键索引)的时候,默认加锁方式是next-key 锁;如果检索条件没有索引,更新数据时会锁住整张表。一个间隙被事务加了锁,其他事务是不能在这个间隙插入记录的,这样可以防止幻读;(避免幻读)

    3、事务隔离级别为串行化时,同一时间将不会再允许其他事务型的操作。

    4、隔离级别越高,越能保证数据的完整性和一致性,但是对并发性能的影响也越大。

    不同的隔离级别有不同的现象,并有不同的锁和并发机制,隔离级别越高,数据库的并发性能就越差,下图为4种事隔离级别与并发性能的关系:

  • 相关阅读:
    今日成长笔记2016-11-18
    牛人博客
    c 、c++、java区别
    Java开发中的23种设计模式详解
    JAVA编程规范
    设计及编码质量改进之降低耦合度
    加密
    敏捷开发之Scrum扫盲篇
    RPC
    李洪强iOS开发Swift篇—04_运算符
  • 原文地址:https://www.cnblogs.com/xs-yqz/p/13366758.html
Copyright © 2011-2022 走看看