首页 > 数据库 >45. SQL--事务(非常详细)

45. SQL--事务(非常详细)

时间:2022-09-05 16:23:11浏览次数:72  
标签:语句 事务 SQL -- 数据库 45 回滚 sql mysql

1. 前言

在数据库中,我们将一条 sql 语句称为一次基本的操作。将若干条 sql 语句“打包”在一起,共同执行一个完整的任务,这就是事务。

事务( transaction)由一次或者多次基本操作构成,或者说,事务由一条或者多条 sql 语句构成。

事务有一个最显著的特征,就是它包含的所有 sql 语句作为一个整体向数据库提交,只有所有的 sql 语句都执行完成,整个事务才算成功,一旦某个 sql 语句执行失败,整个事务就失败了。事务失败后需要回滚所有的 sql 语句。

事务中的所有 sql 语句是一个整体,共同进退,不可分割,要么全部执行成功,要么全部执行失败。

事务有很多实用的场景。例如对于电商网站,通常将用户订单存储在一张表中,将商品库存情况存储在另一张表中,当有用户下单时,需要执行两条 sql 语句,一条负责更新订单表,一条负责更新库存表,这两条 sql 语句必须同时执行成功。如果只有一条语句执行成功,另一条语句执行失败,将导致数据库出错,这种后果是无法接受的。

为了避免出现意外,可以将以上两条语句放到一个事务中,其中一条语句执行失败时,数据库将回滚到原来的状态。对于买家来说,数据库回滚会导致下单失败,但这很容易处理,让买家再次下单即可。数据库的正确性永远是最重要的。

其实我们平时使用数据库时,就已经在使用事务了,只不过这种事务只包含一条 sql 语句,并且由数据库引擎自动封装和提交。这意味着,对于任何一条 sql 语句,要么执行成功,要么执行失败,不能成功一部分,失败一部分。

2. 事务的属性

一般来说,事务具有四个标准属性,分别是原子性(atomicity,或称不可分割性)、一致性(consistency)、隔离性(isolation,又称独立性)、持久性(durability),简称 acid。具体说明如下:
1) 原子性
一个事务中的所有 sql 语句,要么全部执行成功,要么全部执行失败,不会结束在中间的某个环节。事务在执行过程中发生错误,会被回滚(rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。
2) 一致性
在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的数据必须完全符合所有的预设规则,其中包含数据的精确度、串联性以及后续数据库可以自发性地完成预定的工作。
3) 隔离性
数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。事务隔离分为不同级别,包括读未提交(read uncommitted)、读提交(read committed)、可重复读(repeatable read)和串行化(serializable)。
4) 持久性
事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失。

3. 事务执行流程

各种数据库对事务的支持细节不尽相同,本教程以 mysql 数据库为例进行讲解,mysql 使用标准 sql 来支持事务。

与事务控制有关的 sql 命令包括:

  • begin 或者 start transaction:开始事务;
  • commit:提交事务;
  • rollback:回滚事务;
  • savepoint:在事务内部设置回滚标记点;
  • release savepoint:删除回滚标记点;
  • rollback to:将事务回滚到标记点(rollback 命令的一种变形写法)。

一个事务要么提交(commit),要么回滚(rollback),提交意味着成功,回滚意味着失败。编写事务代码时,以 begin 命令开头,后跟一条或者多条 sql 语句,最后书写 commit 或者 rollback 命令;commit 和 rollback 对应事务的两种状态,只能出现一个。

事务控制命令仅能与 dml 类别的 sql 命令一起使用,包括 insert、update、delete 和 select,在创建或者删除表时不能使用事务,因为这些操作在数据库中是自动提交的。

1) 开始事务

开始事务有以下两种命令,选择其一即可:

commit;

或者

start transaction;

该命令用来标记一个事务的起始点。

2) 提交事务

提交事务使用如下命令:

commit;

提交事务意味着真正执行事务包含的 sql 语句,并把对数据库的修改写入到磁盘上的物理数据库中。commit 意味着事务结束,并且执行成功。

例如,有包含如下记录的 customers 表:

+----+----------+-----+-----------+----------+
| ID | NAME     | AGE | ADDRESS   | SALARY   |
+----+----------+-----+-----------+----------+
|  1 | Ramesh   |  32 | Ahmedabad |  2000.00 |
|  2 | Khilan   |  25 | Delhi     |  1500.00 |
|  3 | kaushik  |  23 | Kota      |  2000.00 |
+----+----------+-----+-----------+----------+

打开一个 MySQL 命令行窗口(我们称它为 A 窗口),使用事务向表中插入两条数据:

mysql> begin;
mysql> insert into customers (name, age, address, salary) values ('chaitali', 25, 'mumbai', 6500.00 );
mysql> insert into customers (name, age, address, salary) values ('hardik', 27, 'bhopal', 8500.00 );

再打开另外一个 mysql 命令行窗口(我们称它为 b 窗口),使用 select 命令查看 customers 表的内容:

mysql>  SELECT * FROM CUSTOMERS;
+----+---------+-----+-----------+--------+
| id | name    | age | address   | salary |
+----+---------+-----+-----------+--------+
|  1 | Ramesh  |  32 | Ahmedabad |   2000 |
|  2 | Khilan  |  25 | Delhi     |   1500 |
|  3 | Kaushik |  23 | Kota      |   2000 |
+----+---------+-----+-----------+--------+

您看,A 窗口对表的修改并没有影响到 B 窗口,B 窗口只能查看到修改之前的内容,这说明 A 窗口的修改动作并没有立即更新到物理数据库,所以在其它会话窗口中无法看到。

在 A 窗口中提交事务:

mysql> commit;

在 b 窗口中再次查看 customers 表的内容:

mysql>  SELECT * FROM CUSTOMERS;
+----+----------+-----+-----------+--------+
| id | name     | age | address   | salary |
+----+----------+-----+-----------+--------+
|  1 | Ramesh   |  32 | Ahmedabad |   2000 |
|  2 | Khilan   |  25 | Delhi     |   1500 |
|  3 | Kaushik  |  23 | Kota      |   2000 |
|  4 | Chaitali |  25 | Mumbai    |   6500 |
|  5 | Hardik   |  27 | Bhopal    |   8500 |
+----+----------+-----+-----------+--------+

您看,在 A 窗口提交事务以后,对表所做的修改才真正更新到物理数据库中,所以才能在其它会话窗口中查看到 A 窗口中插入的数据。

3) 回滚事务

回滚意味着撤销尚未保存到物理数据库中的操作,具体语法格式如下:

rollback;

事务执行过程中如果发生了某种故障,事务不能继续执行,就可以撤销事务,此时对数据库的修改并不会保存到物理数据库中。撤销意味着事务结束,并且执行失败。

例如,有包含如下记录的 CUSTOMERS 表:

+----+----------+-----+-----------+--------+
| id | name     | age | address   | salary |
+----+----------+-----+-----------+--------+
|  1 | Ramesh   |  32 | Ahmedabad |   2000 |
|  2 | Khilan   |  25 | Delhi     |   1500 |
|  3 | Kaushik  |  23 | Kota      |   2000 |
|  4 | Chaitali |  25 | Mumbai    |   6500 |
|  5 | Hardik   |  27 | Bhopal    |   8500 |
+----+----------+-----+-----------+--------+

使用事务删除最后两个用户,并回滚:

mysql> BEGIN;
mysql> DELETE FROM CUSTOMERS WHERE ID=4;
mysql> DELETE FROM CUSTOMERS WHERE ID=5;
mysql> ROLLBACK;

mysql> SELECT * FROM CUSTOMERS;
+----+----------+-----+-----------+--------+
| id | name     | age | address   | salary |
+----+----------+-----+-----------+--------+
|  1 | Ramesh   |  32 | Ahmedabad |   2000 |
|  2 | Khilan   |  25 | Delhi     |   1500 |
|  3 | Kaushik  |  23 | Kota      |   2000 |
|  4 | Chaitali |  25 | Mumbai    |   6500 |
|  5 | Hardik   |  27 | Bhopal    |   8500 |
+----+----------+-----+-----------+--------+

您看,回滚事务以后,物理数据库中的数据并没有发生改变,表中的内容和事务执行之前的内容一致。

4) 回滚标记点

ROLLBACK 命令默认回滚整个事务,也即事务中的所有修改操作都无效。但是 SQL 允许回滚事务的一部分,此时需要在事务中设置一个标记点,在该标记点之后的 SQL 语句将被回滚,之前的 SQL 语句将被成功执行。

设置标记点使用 SAVEPOINT 命令,具体语法如下:

SAVEPOINT point_name;

point_name 为标记点名字。

回滚到标记点使用 ROLLBACK TO 命令,具体语法如下:

ROLLBACK TO point_name;

例如,有包含如下记录的 CUSTOMERS 表:

+----+----------+-----+-----------+--------+
| id | name     | age | address   | salary |
+----+----------+-----+-----------+--------+
|  1 | Ramesh   |  32 | Ahmedabad |   2000 |
|  2 | Khilan   |  25 | Delhi     |   1500 |
|  3 | Kaushik  |  23 | Kota      |   2000 |
|  4 | Chaitali |  25 | Mumbai    |   6500 |
|  5 | Hardik   |  27 | Bhopal    |   8500 |
+----+----------+-----+-----------+--------+

使用事务删除最后两个用户,并回滚到标记点:

mysql> BEGIN;
mysql> DELETE FROM CUSTOMERS WHERE ID=4;
mysql> SAVEPOINT sp;
mysql> DELETE FROM CUSTOMERS WHERE ID=5;
mysql> ROLLBACK TO sp;

mysql> SELECT * FROM CUSTOMERS;
+----+---------+-----+-----------+--------+
| id | name    | age | address   | salary |
+----+---------+-----+-----------+--------+
|  1 | Ramesh  |  32 | Ahmedabad |   2000 |
|  2 | Khilan  |  25 | Delhi     |   1500 |
|  3 | Kaushik |  23 | Kota      |   2000 |
|  5 | Hardik  |  27 | Bhopal    |   8500 |
+----+---------+-----+-----------+--------+

您看,我们回滚到标记点 sp,只有 ID 为 4 的用户被删除,ID 为 5 的用户依然留在数据库中。

标签:语句,事务,SQL,--,数据库,45,回滚,sql,mysql
From: https://www.cnblogs.com/jiajunling/p/16658579.html

相关文章

  • php 中 global的问题
    在教学中就跟做软件测试一样,分发现很多新大陆,因为你不停的去做各种尝试,也因为书本或者是教程或者是帮助文档其实说的也不是太清楚,以前也忘了总结,现在就对今天发现的这个问......
  • # JS输入输出语句
    JS输入输出语句方法说明归属alert(msg)浏览器弹出警示框浏览器console.log(msg)浏览器控制台打印输出信息浏览器prompt(info)浏览器弹出输出框,用户......
  • 计算机科学速成课第一课:计算机早期历史
    课程链接:https://www.bilibili.com/video/av21376839?p=3第一课:计算机早期历史0、课程目标:从高层次总览一系列计算机话题,快速入门计算机科学。1、计算机技术的影响——......
  • 物理不可克隆函数-PUF
     时间:2022/09/05 一.简介PUF技术是一组微型延迟电路,其通过提取芯片制造过程中的差异,生成无数个独特的、不可预测的“密钥”。输入也称为激励(Challenge),输出也称为......
  • Jenkins持续集成.NET项目(SVN+MSBuild)
    一、安装Jenkins和MSBuild工具官网下载最新版Jenkins (Jenkins运行需要JDK环境,需要提前配置JDK1.8及以上版本),安装完成后,在浏览器打开http://localhost:8080/(默认端口为8......
  • 题解【CF1316E Team Building】
    题目传送门状压DP入门题。设\(f_{i,S}\)表示考虑了前\(i\)个人,队伍放置情况为\(S\)时(0表示放置了队员,1表示没有放置)的最大贡献。然后分讨一下\(i\)是去当队......
  • Java学习-第一部分-第二阶段-第八节:IO流
    IO流笔记目录:(https://www.cnblogs.com/wenjie2000/p/16378441.html)IO流体系图文件什么是文件文件.对我们并不陌生,文件是保存数据的地方,比如大家经常使用的word文......
  • react向路由组件传递参数数据的3种方式
    1、params传递参数步骤:(1)路由链接(携带参数)<Linkto={`/home/message/detail/${ele.id}/${ele.title}`}>{ele.title}</Link>(2)注册路由(声明接收):<Routepath='/home......
  • 今日内容 视图层与模板层
    网页伪静态实际上伪静态是个动态页面,只是通过技术手段伪装成立静态页面的样子,伪静态页面的内容是通过读取数据库生成的。将动态网页伪装成静态网页从而提升网页被搜......
  • 【校招VIP】[前端][二本][6分]简历的板式比较标准
    关注【校招VIP】公众号,回复【简历】,添加校招顾问微信,即可获取简历指导!本份简历是一位21届二本前端同学的简历,简历评分6分。一、学员简历二、指导意见简历的版式没有问......