事务

什么是事务?

事务就是一组原子性的SQL查询,或者说是一个独立的工作单元。

事务的作用

事务在我们平常的CRUD(增删改查)操作当中也许不太常用, 但是如果我们有一种需求,一组操作中必须全部成功执行,才算完成任务,只要有一个出错了,那么所有的任务都将回到最初的状况,恢复原样。那么这就需要使用事务了。如: 银行转账,购买飞机票......

事务的特性

事务具有4个特性(ACID):原子性(Atomicity)、一致性(Consistency)、隔离性(lsolation)、持久性(Durability)。

  • 原子性(Atomicity): 事务中的逻辑要全部执行,不可分割。(原子是物理中最小单位)
  • 一致性(Consistency): 指事务执行前和执行后, 数据的完整性保持一致
  • 隔离性(Isolation): 指一个事务在执行的过程中不应该受其他事务的影响
  • 持久性(Durability): 事务执行结束(提交或回滚), 数据都应持久化到数据中

演示事务

在MySQL命令行的默认设置下,事务都是自动提交的,即执行SQL语句后就会马上执行commit操作。因此要显式地开启一个事务务须使用命令begin或start transaction,或者执行命令set autocommit = 0,用来禁止使用当前会话的自动提交。

  1. 命令行中演示事务

    开启事务:start transaction;

    提交事务:commit;  数据将会写到磁盘上的数据库

    回滚事务:rollback;  数据回滚,回到最初的状态。

    开启一个事物:

    我们先看看stuifo表里最初的数据:

    再更新stuifo表里的派大星的性别为女并查看修改后的数据:

    这里显示性别已经修改了,但其实数据没有真正的写入到硬盘中,我们去MySQL管理软件看看数据有没有被更新:

    可以发现数据没有被更新,因为事务没有被提交,现在我们把事务提交。

    提交后,数据才会被更新,真正的写入到硬盘里。

    如果不提交,使用回滚rollback,数据就会恢复到初始状态:

  2. 使用代码演示事务

    1. package test;
    2. import Utils.JDBCUtils;
    3. import java.sql.Connection;
    4. import java.sql.PreparedStatement;
    5. import java.sql.SQLException;
    6. public class demo {
    7. public static void main(String[] args) {
    8. Connection conn = null;
    9. PreparedStatement ps = null;
    10. try {
    11. conn = JDBCUtils.getConnection();
    12. conn.setAutoCommit(false);//关闭事务默认自动提交
    13. String sql = "update stuifo set sex = '女' where id = 2";
    14. ps = conn.prepareStatement(sql);
    15. ps.executeUpdate();
    16. conn.commit();//成功,提交事务
    17. } catch (SQLException e) {
    18. try {
    19. conn.rollback();//失败,回滚事务
    20. } catch (SQLException e1) {
    21. e1.printStackTrace();
    22. }
    23. e.printStackTrace();
    24. } finally {
    25. JDBCUtils.closeStatement(ps);
    26. JDBCUtils.closeConn(conn);
    27. }
    28. }
    29. }

安全问题

读问题

在操作数据库时,可能会出现读问题,主要表现为脏读,不可重复读,幻读。

  1. 脏读

    脏读: 指 一个事务读到了另一个事务还未提交的数据。

    演示:

    最初stuifo表中的数据为:

    在两个窗口分别开启一个事务,首先在A窗口修改派大星性别为女,但不提交事务,然后在B窗口中查询表stuifo。

    A窗口:

    B窗口:

    这样B窗口中的事务就读取到了A窗口中的事务未提交的数据,称之为脏读。

  2. 不可重复读

    不可重复读: 一个事务读到了另一个事务提交的数据, 导致多次查询结果不一致。

    最初stuifo表中的数据为:

    mysql默认是可重复读的,所以我们先把A窗口的隔离级别设置为读未提交,在两个窗口分别开启一个事物,首先在A窗口查询表stuifo,然后在B窗口中修改派大星性别为女,并提交事务,再在A窗口查询表stuifo。

    A窗口:

    B窗口:

    A窗口:

    这样A窗口中在一个事务下前后读取的数据不一致,称为不可重复读。

  3. 幻读

    幻读: 一个事务读到了另一个事务已提交的插入的数据,导致多次查询结果不一致。

    幻读与不可重复读类似,都是前后查询结果不一致,但是不可重复读针对的是update,后者针对的是insert。

写问题

在操作数据库时,可能会出现写问题,主要表现为丢失更新。

丢失更新:指一个事务去修改数据库,另一个事务也修改数据库,最后的那个事务,不管是提交还是回滚都会造成前面一个事务的数据更新丢失。

解决丢失更新,通常有两种方法:悲观锁和乐观锁。

  • 悲观锁  

    指事务在一开始就认为丢失更新一定会发生,这是一件很悲观的事情。具体操作步骤如下:

    1.所有事务在执行操作前,先查询一次数据,查询语句如下:

    select * from student for update ; 后面的for update其实是数据库锁机制 、一种排他锁。

    2.哪个事务先执行这个语句,哪个事务就持有了这把锁,可以查询出来数据,后面的事务再执行这条语句,不会有任何数据显示,就只能等着。

    3.一直等到前面的那个事务提交数据后,后面的事务数据才会出来,那么才可以往下接着操作。

这相当于上卫生间似的,谁先来谁就进去蹲着,后面来的人都只能等着。只有里面的人出来了,外面的人才能进去。

  • 乐观锁

    乐观锁是指,从来不会觉得丢失更新会发生。那么它的具体做法是什么呢?

    要求程序员在数据库中添加字段,然后在后续更新的时候,对该字段进行判定比对,如果一致才允许更新。例子如下:

    1.数据库表中,额外添加了一个version字段,用于记录版本,默认从0开始,只要有针对表中数据进行修改的,那么version就+1。

    2.开启A事务,然后开启B事务。

    3.先执行数据库表操作。因为以前都没有人修改过。所以是允许A事务修改数据库的,但是修改完毕,就把version的值变成1了。

    4.B事务,这时候如果想执行修改,那么是不允许修改的。因为B事务以前是没有查询过数据库内容的,所以它认为数据库版本还是0。但是数据库的版本经过A修改,已经是1了。所以这时候不允许修改,要求其重新查询 。

    5.B重新查询后,将会得到version为1的数据,这份数据就是之前A事务修改的数据了,B在进行修改,也是在A的基础上修改的。所以就不会有丢失更新的情况出现了。

隔离级别

设置事务的隔离级别可以解决上面的脏读,不可重复读,幻读问题,但是隔离级别越高,执行效率越低,mysql的隔离级别默认为可重复读。

事务隔离分为四个级别,由低到高为读未提交(Read uncommitted)、读提交(read committed)、可重复读(repeatable read)和串行化(Serializable)。

✔代表可能发生,✖代表不会发生

隔离级别 脏读 不可重复读 幻读
读未提交(Read uncommitted)
读提交(read committed)
可重复读(repeatable read)
串行化(Serializable)

查看隔离级别:

设置隔离级别:

Java Web(八) 事务,安全问题及隔离级别的更多相关文章

  1. 数据库事务的四大特性以及事务的隔离级别-与-Spring事务传播机制&隔离级别

    数据库事务的四大特性以及事务的隔离级别   本篇讲诉数据库中事务的四大特性(ACID),并且将会详细地说明事务的隔离级别. 如果一个数据库声称支持事务的操作,那么该数据库必须要具备以下四个特性: ⑴ ...

  2. Spring事务传播机制&隔离级别

    一.Propagation (事务的传播属性) Propagation : key属性确定代理应该给哪个方法增加事务行为.这样的属性最重要的部份是传播行为.有以下选项可供使用:PROPAGATION_ ...

  3. 数据库事务中的隔离级别和锁+spring Transactional注解

    数据库事务中的隔离级别和锁 数据库事务在后端开发中占非常重要的地位,如何确保数据读取的正确性.安全性也是我们需要研究的问题.ACID首先总结一下数据库事务正确执行的四个要素(ACID): 原子性(At ...

  4. mysql事务之间的隔离级别

    事务间未做隔离,会引起下面这些问题. 1.脏读:一个事务可读到另外一个尚未commit的事务中的数据. 2.不可重复读:在一个事务中,读取同一个数据 a,b,按顺序读取,在读a  b 之间,另外一个事 ...

  5. [转]数据库事务中的隔离级别和锁+spring Transactional注解

    数据库事务中的隔离级别和锁 数据库事务在后端开发中占非常重要的地位,如何确保数据读取的正确性.安全性也是我们需要研究的问题.ACID首先总结一下数据库事务正确执行的四个要素(ACID): 原子性(At ...

  6. 一文彻底读懂MySQL事务的四大隔离级别

    前言 之前分析一个死锁问题,发现自己对数据库隔离级别理解还不够清楚,所以趁着这几天假期,整理一下MySQL事务的四大隔离级别相关知识,希望对大家有帮助~ 事务 什么是事务? 事务,由一个有限的数据库操 ...

  7. MYSQL数据库事务4种隔离级别及7种传播行为

    事务的特性: 原子性:事务的不可分割,组成事务的各个逻辑单元不可分割. 一致性:事务执行的前后,数据完整性保持一致. 隔离性:事务执行不应该受到其他事务的干扰. 持久性:事务一旦结束,数据就持久化到数 ...

  8. 数据库事务ACID与隔离级别

    如果一个数据库声称支持事务的操作,那么该数据库必须要具备以下四个特性: 原子性(Atomicity) 原子性是指事务包含的所有操作要么全部成功,要么全部失败回滚,这和前面两篇博客介绍事务的功能是一样的 ...

  9. mysql事务四种隔离级别

    事务的基本要素:原子性,一致性,隔离性,持久性. 事务并发问题:脏读,不可重复读,幻读. mysql隔离级别:read-uncommitted,read-committed,repeatable-re ...

随机推荐

  1. 给 vue项目添加ESLint

    eslint配置方式有两种: 注释配置:使用js注释来直接嵌入ESLint配置信息到一个文件里 配置文件:使用一个js,JSON或者YAML文件来给整个目录和它的子目录指定配置信息.这些配置可以写在一 ...

  2. 清理SuperMap iclient 三维插件的缓存批处理

    在windows任何位置,新建一个文本文件,将下面内容复制到文本文件中并保存,然后修改该文本文件后缀为.bat,鼠标点击执行即可完成清理工作~ @echo off title 清理三维缓存 @echo ...

  3. Python splinter 环境搭建

    今天无意间看到了splinter. Splinter是一个使用Python开发的开源Web应用测试工具.它可以帮你实现自动浏览站点和与其进行交互. Splinter对已有的自动化工具(如:Seleni ...

  4. HeadFirst Ruby 第七章总结 references

    前言 这一章的内容关于 references,讲了当 Ruby 程序中可能会遇到关于 reference 与 object 之间概念混淆而导致的问题. 导言 本章从一个 astronomer 发现 s ...

  5. 雷林鹏分享:C# 数组(Array)

    C# 数组(Array) 数组是一个存储相同类型元素的固定大小的顺序集合.数组是用来存储数据的集合,通常认为数组是一个同一类型变量的集合. 声明数组变量并不是声明 number0.number1... ...

  6. tornado web

    tornado web frame: 非阻塞服务器,速度快,运用epoll 模板语言+render(),实现根据用户输入,自动渲染页面的动态效果. 在使用模板前需要在setting中设置模板路径: s ...

  7. 搭建智能合约开发环境Remix IDE及使用

    目前开发智能的IDE, 首推还是Remix, 而Remix官网, 总是由于各种各样的(网络)原因无法使用,本文就来介绍一下如何在本地搭建智能合约开发环境remix-ide并介绍Remix的使用. 写在 ...

  8. 时钟中断TIMER_BH(bottom_half)实现分析

    017-12-6 16:27:35时钟中断TIMER_BH(bottom_half)实现分析1.3.1001. 时钟0号中断安装    setup_x86_irq(0, &irq0);@arc ...

  9. hdu-3001 三进制状态压缩+dp

    用dp来求最短路,虽然效率低,但是状态的概念方便解决最短路问题中的很多限制,也便于压缩以保存更多信息. 本题要求访问全图,且每个节点不能访问两次以上.所以用一个三进制数保存全图的访问状态(3^10,空 ...

  10. php数组的逐行写入文件与读取

    <?php /** * * 对数组$arr1=['Apple Orange Banana Strawberry'] 写入文件,并读取 **/ class IoFile { private $pa ...