likes
comments
collection
share

【JavaWeb】MySQL高级篇之事务

作者站长头像
站长
· 阅读数 13

事务

1 - 概述

  • 数据库的事务(Transaction)是一种机制、一个操作序列,包含了一组数据库操作命令

  • 事务把所有的命令作为一个整体一起向系统提交或撤销操作请求,即这一组数据库命令要么同时成功,要么同时失败

  • 事务是一个不可分割的工作逻辑单元

这些概念可能不太好理解,接下来我们举例来进行说明,如下图有一张表

【JavaWeb】MySQL高级篇之事务

张三和李四账户中各有 1000 块钱,现李四需要转账 500 块钱给张三,具体的转账操作为

  • 第一步:查询李四账户余额
  • 第二步:从李四账户金额 -500
  • 第三步:给张三账户金额 +500

现在假设在转账过程中,第二步完成后出现了异常导致第三步没有执行,这样就会造成李四账户金额少了 500,而张三账户金额并没有多 500;这样的系统是有问题的。

怎么解决呢?其实我们使用事务就可以解决上述问题

【JavaWeb】MySQL高级篇之事务

从上图可以看到在转账前开启事务,如果出现了异常就回滚事务,第三步正常执行就提交事务,这样就可以完美解决问题

2 - 语法

  • 开启事务

    START TRANSACTION;
    或者  
    BEGIN;
    
  • 提交事务

    commit;
    
  • 回滚事务

    rollback;
    

3 - 代码验证

  • 环境准备

    DROP TABLE IF EXISTS account;
    
    -- 创建账户表
    CREATE TABLE account(
    	id int PRIMARY KEY auto_increment,
    	name varchar(10),
    	money double(10,2)
    );
    
    -- 添加数据
    INSERT INTO account(name,money) values('张三',1000),('李四',1000);
    
  • 不加事务演示问题

    -- 转账操作
    -- 1. 查询李四账户金额是否大于500
    
    -- 2. 李四账户 -500
    UPDATE account set money = money - 500 where name = '李四';
    
    出现异常了...  -- 此处不是注释,在整体执行时会出问题,后面的sql则不执行
    -- 3. 张三账户 +500
    UPDATE account set money = money + 500 where name = '张三';
    

    整体执行结果肯定会出问题,我们查询账户表中数据,发现李四账户少了 500

    【JavaWeb】MySQL高级篇之事务

  • 添加事务 sql

    -- 开启事务
    BEGIN;
    -- 转账操作
    -- 1. 查询李四账户金额是否大于500
    
    -- 2. 李四账户 -500
    UPDATE account set money = money - 500 where name = '李四';
    
    出现异常了...  -- 此处不是注释,在整体执行时会出问题,后面的sql则不执行
    -- 3. 张三账户 +500
    UPDATE account set money = money + 500 where name = '张三';
    
    -- 提交事务
    COMMIT;
    
    -- 回滚事务
    ROLLBACK;
    

    上面 sql 中的执行成功进选择执行提交事务,而出现问题则执行回滚事务的语句。以后我们肯定不可能这样操作,而是在 java 中进行操作,在 java 中可以抓取异常,没出现异常提交事务,出现异常回滚事务

4 - 事务的四大特征

  • 原子性(Atomicity):事务是不可分割的最小操作单位,要么同时成功,要么同时失败
  • 一致性(Consistency):事务完成时,必须使所有的数据都保持一致状态
  • 隔离性(Isolation):多个事务之间,操作的可见性
  • 持久性(Durability):事务一旦提交或回滚,它对数据库中的数据的改变就是永久的

说明:

mysql 中事务是自动提交的,也就是说我们不添加事务执行 sql 语句,语句执行完毕会自动的提交事务   我们可以通过下面语句查询默认提交方式:

SELECT @@autocommit;

查询到的结果是1 则表示自动提交,结果是 0 表示手动提交。当然也可以通过下面语句修改提交方式

set @@autocommit = 0;