什么是 SQL 事务,如何创建 SQL 事务
liuian 2025-04-05 20:09 27 浏览
目录
- 一、什么是事务
- 二、创建事务
- 三、ACID 特性
本文给大家介绍数据库中用来管理数据更新的重要概念——SQL 事务。简单来讲,事务就是需要在同一个处理单元中执行的一系列更新处理的集合。
本文重点
事务是需要在同一个处理单元中执行的一系列更新处理的集合。通过使用事务,可以对数据库中的数据更新处理的提交和取消进行管理。
事务处理的终止指令包括 COMMIT(提交处理)和 ROLLBACK(取消处理)两种。
DBMS 的事务具有原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)和持久性(Durability)四种特性。通常将这四种特性的首字母结合起来,统称为 ACID 特性。
一、什么是事务
估计有些读者对事务(transaction)这个词并不熟悉,它通常被用于商务贸易或者经济活动中,但是在 RDBMS 中,事务是对表中数据进行更新的单位。简单来讲,事务就是需要在同一个处理单元中执行的一系列更新处理的集合。
如 SQL 如何插入、删除和更新数据 所述,对表进行更新需要使用 INSERT、DELETE 或者 UPDATE 三种语句。
但通常情况下,更新处理并不是执行一次就结束了,而是需要执行一系列连续的操作。这时,事务就能体现出它的价值了。
说到事务的例子,请大家思考一下下述情况。
现在,请大家把自己想象为管理 Product(商品)表的程序员或者软件工程师。销售部门的领导对你提出了如下要求。
“某某,经会议讨论,我们决定把 运动 T 恤 的销售单价下调 1000 元,同时把 T 恤衫 的销售单价上浮 1000 元,麻烦你去更新一下数据库。”
由于大家已经学习了更新数据的方法——只需要使用 UPDATE 进行更新就可以了,所以肯定会直接回答“知道了,请您放心吧”。
此时的事务由如下两条更新处理所组成。
- 更新商品信息的事务
- ① 将 运动 T 恤 的销售单价降低 1000 元
- UPDATE ProductSET sale_price = sale_price - 1000WHERE product_name = '运动T恤';
- ② 将 T 恤衫 的销售单价上浮 1000 元
- UPDATE ProductSET sale_price = sale_price + 1000WHERE product_name = 'T恤衫';
上述 ① 和 ② 的操作一定要作为同一个处理单元执行。
如果只执行了 ① 的操作而忘记了执行 ② 的操作,或者反过来只执行了 ② 的操作而忘记了执行 ① 的操作,一定会受到领导的严厉批评。
遇到这种需要在同一个处理单元中执行一系列更新操作的情况,一定要使用事务来进行处理。
法则 7
事务是需要在同一个处理单元中执行的一系列更新处理的集合。
一个事务中包含多少个更新处理或者包含哪些处理,在 DBMS 中并没有固定的标准,而是根据用户的要求决定的(例如,运动 T 恤 和 T 恤衫 的销售单价需要同时更新这样的要求,DBMS 是无法了解的)。
二、创建事务
如果想在 DBMS 中创建事务,可以按照如下语法结构编写 SQL 语句。
语法 6 事务的语法
事务开始语句; DML语句①; DML语句②; DML语句③; . . .事务结束语句(COMMIT或者ROLLBACK);
使用事务开始语句和事务结束语句,将一系列 DML 语句(INSERT/UPDATE/DELETE 语句)括起来,就实现了一个事务处理。
这时需要特别注意的是事务的开始语句 [1]。实际上,在标准 SQL 中并没有定义事务的开始语句,而是由各个 DBMS 自己来定义的。比较有代表性的语法如下所示。
- SQL Server、PostgreSQL
- BEGIN TRANSACTION
- MySQL
- START TRANSACTION
- Oracle、DB2
- 无
例如使用之前的那两个 UPDATE(① 和 ②)创建出的事务如代码清单 21 所示。
代码清单 21 更新商品信息的事务
SQL Server PostgreSQL
BEGIN TRANSACTION; -- 将运动T恤的销售单价降低1000日元 UPDATE Product SET sale_price = sale_price - 1000 WHERE product_name = '运动T恤'; -- 将T恤衫的销售单价上浮1000日元 UPDATE Product SET sale_price = sale_price + 1000 WHERE product_name = 'T恤衫'; COMMIT;
MySQL
START TRANSACTION; -- 将运动T恤的销售单价降低1000日元 UPDATE Product SET sale_price = sale_price - 1000 WHERE product_name = '运动T恤'; -- 将T恤衫的销售单价上浮1000日元 UPDATE Product SET sale_price = sale_price + 1000 WHERE product_name = 'T恤衫'; COMMIT;
Oracle DB2
-- 将运动T恤的销售单价降低1000日元UPDATE Product SET sale_price = sale_price - 1000 WHERE product_name = '运动T恤'; -- 将T恤衫的销售单价上浮1000日元UPDATE Product SET sale_price = sale_price + 1000 WHERE product_name = 'T恤衫'; COMMIT;
如上所示,各个 DBMS 事务的开始语句都不尽相同,其中 Oracle 和 DB2 并没有定义特定的开始语句。
可能大家觉得这样的设计很巧妙,其实是因为标准 SQL 中规定了一种悄悄开始事务处理 [2] 的方法。
因此,即使是经验丰富的工程师也经常会忽略事务处理开始的时间点。大家可以试着通过询问“是否知道某个 DBMS 中事务是什么时候开始的”,来测试学校或者公司前辈的数据库知识。
反之,事务的结束需要用户明确地给出指示。结束事务的指令有如下两种。
- COMMIT——提交处理
- COMMIT 是提交事务包含的全部更新处理的结束指令(图 3),相当于文件处理中的覆盖保存。一旦提交,就无法恢复到事务开始前的状态了。
- 因此,在提交之前一定要确认是否真的需要进行这些更新。
- 图 3 COMMIT 的流程 = 直线进行
- 万一由于误操作提交了包含错误更新的事务,就只能回到重新建表、重新插入数据这样繁琐的老路上了。
- 由于可能会造成数据无法恢复的后果,请大家一定要注意(特别是在执行 DELETE 语句的 COMMIT 时尤其要小心)。
- 法则 8
- 虽然我们可以不清楚事务开始的时间点,但是在事务结束时一定要仔细进行确认。
- ROLLBACK——取消处理
- ROLLBACK 是取消事务包含的全部更新处理的结束指令(图 4),相当于文件处理中的放弃保存。一旦回滚,数据库就会恢复到事务开始之前的状态(代码清单 22)。
- 通常回滚并不会像提交那样造成大规模的数据损失。
- 图 4 ROLLBACK 的流程 = 掉头回到起点
- 代码清单 22 事务回滚的例子
- SQL Server PostgreSQL
- BEGIN TRANSACTION; ------------------- ① -- 将运动T恤的销售单价降低1000日元 UPDATE Product SET sale_price = sale_price - 1000 WHERE product_name = '运动T恤'; -- 将T恤衫的销售单价上浮1000日元 UPDATE Product SET sale_price = sale_price + 1000 WHERE product_name = 'T恤衫'; ROLLBACK;
- 特定的 SQL
- 至此,我们已经知道各个 DBMS 中关于事务的语法不尽相同。
- 代码清单 22 中的语句在 MySQL 中执行时需要将 ① 语句改写为“START TRANSACTION”,而在 Oracle 和 DB2 中执行时则无需 ① 语句(请将其删除),具体请参考上一节的“创建事务”。
- 上述事务处理执行之后,表中的数据不会发生任何改变。这是因为执行最后一行的 ROLLBACK 之后,所有的处理都被取消了。
- 因此,回滚执行起来就无需像提交时那样小心翼翼了(即使是想要提交的情况,也只需要重新执行事务处理就可以了)。
专栏
事务处理何时开始
之前我们说过,事务并没有标准的开始指令存在,而是根据 DBMS 的不同而不同。
实际上,几乎所有的数据库产品的事务都无需开始指令。这是因为大部分情况下,事务在数据库连接建立时就已经悄悄开始了,并不需要用户再明确发出开始指令。
例如,使用 Oracle 时,数据库连接建立之后,第一条 SQL 语句执行的同时,事务就已经悄悄开始了。
像这样不使用指令而悄悄开始事务的情况下,应该如何区分各个事务呢?通常会有如下两种情况。
A:每条 SQL 语句就是一个事务(自动提交模式)
B:直到用户执行 COMMIT 或者 ROLLBACK 为止算作一个事务
通常的 DBMS 都可以选择其中任意一种模式。默认使用自动提交模式的 DBMS 有 SQL Server、PostgreSQL 和 MySQL 等 13 DML 语句如下所示,每一条语句都括在事务的开始语句和结束语句之中。
BEGIN TRANSACTION; -- 将运动T恤的销售单价降低1000日元 UPDATE Product SET sale_price = sale_price - 1000 WHERE product_name = '运动T恤';COMMIT; BEGIN TRANSACTION; -- 将T恤衫的销售单价上浮1000日元 UPDATE Product SET sale_price = sale_price + 1000 WHERE product_name = 'T恤衫';COMMIT;
在默认使用 B 模式的 Oracle 中,事务都是直到用户自己执行提交或者回滚指令才会结束。
自动提交的情况需要特别注意的是 DELETE 语句。
如果不是自动提交,即使使用 DELETE 语句删除了数据表,也可以通过 ROLLBACK 命令取消该事务的处理,恢复表中的数据。
但这仅限于明示开始事务,或者关闭自动提交的情况。如果不小心在自动提交模式下执行了 DELETE 操作,即使再回滚也无济于事了。
这是一个很严重的问题,初学者难免会碰到这样的麻烦。一旦误删了数据,如果无法重新插入,是不是想哭的心都有了?所以一定要特别小心。
三、ACID 特性
DBMS 的事务都遵循四种特性,将这四种特性的首字母结合起来统称为 ACID 特性。这是所有 DBMS 都必须遵守的规则。
- 原子性(Atomicity)
- 原子性是指在事务结束时,其中所包含的更新处理要么全部执行,要么完全不执行,也就是要么占有一切要么一无所有。
- 例如,在之前的例子中,在事务结束时,绝对不可能出现 运动 T 恤 的价格下降了,而 T 恤衫 的价格却没有上涨的情况。
- 该事务的结束状态,要么是两者都执行了(COMMIT),要么是两者都未执行(ROLLBACK)。
- 从事务中途停止的角度去考虑,就能比较容易理解原子性的重要性了。
- 由于用户在一个事务中定义了两条 UPDATE 语句,DBMS 肯定不会只执行其中一条,否则就会对业务处理造成影响。
- 一致性(Consistency)
- 一致性指的是事务中包含的处理要满足数据库提前设置的约束,如主键约束或者 NOT NULL 约束等。
- 例如,设置了 NOT NULL 约束的列是不能更新为 NULL 的,试图插入违反主键约束的记录就会出错,无法执行。
- 对事务来说,这些不合法的 SQL 会被回滚。也就是说,这些 SQL 处理会被取消,不会执行。
- 一致性也称为完整性(图 5)。
- 图 5 保持完整性的流程
- 隔离性(Isolation)
- 隔离性指的是保证不同事务之间互不干扰的特性。该特性保证了事务之间不会互相嵌套。此外,在某个事务中进行的更改,在该事务结束之前,对其他事务而言是不可见的。
- 因此,即使某个事务向表中添加了记录,在没有提交之前,其他事务也是看不到新添加的记录的。
- 持久性(Durability)
- 持久性也可以称为耐久性,指的是在事务(不论是提交还是回滚)结束后,DBMS 能够保证该时间点的数据状态会被保存的特性。
- 即使由于系统故障导致数据丢失,数据库也一定能通过某种手段进行恢复。
- 如果不能保证持久性,即使是正常提交结束的事务,一旦发生了系统故障,也会导致数据丢失,一切都需要从头再来。
- 保证持久性的方法根据实现的不同而不同,其中最常见的就是将事务的执行记录保存到硬盘等存储介质中(该执行记录称为日志)。
- 当发生故障时,可以通过日志恢复到故障发生前的状态。
- 与之相对,事务结束语句只有 COMMIT 和 ROLLBACK 两种,在所有的 RDBMS 中都是通用的。
- 《标准 SQL 手册修订第 4 版》中的记述:希望大家注意事务默认开始的时间点。没有“BEGIN TRANSACTION”这样明确的开始标志。
相关推荐
- 教你把多个视频合并成一个视频的方法
-
一.情况介绍当你有一个m3u8文件和一个目录,目录中有连续的视频片段,这些片段可以连成一段完整的视频。m3u8文件打开后像这样:m3u8文件,可以理解为播放列表,里面是播放视频片段的顺序。视频片段像这...
- 零代码编程:用kimichat合并一个文件夹下的多个文件
-
一个文件夹里面有很多个srt字幕文件,如何借助kimichat来自动批量合并呢?在kimichat对话框中输入提示词:你是一个Python编程专家,完成如下的编程任务:这个文件夹:D:\downloa...
- Java APT_java APT 生成代码
-
JavaAPT(AnnotationProcessingTool)是一种在Java编译阶段处理注解的工具。APT会在编译阶段扫描源代码中的注解,并根据这些注解生成代码、资源文件或其他输出,...
- Unit Runtime:一键运行 AI 生成的代码,或许将成为你的复制 + 粘贴神器
-
在我们构建了UnitMesh架构之后,以及对应的demo之后,便着手于实现UnitMesh架构。于是,我们就继续开始UnitRuntime,以用于直接运行AI生成的代码。PS:...
- 挣脱臃肿的枷锁:为什么说Vert.x是Java开发者手中的一柄利剑?
-
如果你是一名Java开发者,那么你的职业生涯几乎无法避开Spring。它如同一位德高望重的老国王,统治着企业级应用开发的大片疆土。SpringBoot的约定大于配置、SpringCloud的微服务...
- 五年后,谷歌还在全力以赴发展 Kotlin
-
作者|FredericLardinois译者|Sambodhi策划|Tina自2017年谷歌I/O全球开发者大会上,谷歌首次宣布将Kotlin(JetBrains开发的Ja...
- kotlin和java开发哪个好,优缺点对比
-
Kotlin和Java都是常见的编程语言,它们有各自的优缺点。Kotlin的优点:简洁:Kotlin程序相对于Java程序更简洁,可以减少代码量。安全:Kotlin在类型系统和空值安全...
- 移动端架构模式全景解析:从MVC到MVVM,如何选择最佳设计方案?
-
掌握不同架构模式的精髓,是构建可维护、可测试且高效移动应用的关键。在移动应用开发中,选择合适的软件架构模式对项目的可维护性、可测试性和团队协作效率至关重要。随着应用复杂度的增加,一个良好的架构能够帮助...
- 颜值非常高的XShell替代工具Termora,不一样的使用体验!
-
Termora是一款面向开发者和运维人员的跨平台SSH终端与文件管理工具,支持Windows、macOS及Linux系统,通过一体化界面简化远程服务器管理流程。其核心定位是解决多平台环境下远程连接、文...
- 预处理的底层原理和预处理编译运行异常的解决方案
-
若文章对您有帮助,欢迎关注程序员小迷。助您在编程路上越走越好![Mac-10.7.1LionIntel-based]Q:预处理到底干了什么事情?A:预处理,顾名思义,预先做的处理。源代码中...
- 为“架构”再建个模:如何用代码描述软件架构?
-
在架构治理平台ArchGuard中,为了实现对架构的治理,我们需要代码+模型描述所要处理的内容和数据。所以,在ArchGuard中,我们有了代码的模型、依赖的模型、变更的模型等,剩下的两个...
- 深度解析:Google Gemma 3n —— 移动优先的轻量多模态大模型
-
2025年6月,Google正式发布了Gemma3n,这是一款能够在2GB内存环境下运行的轻量级多模态大模型。它延续了Gemma家族的开源基因,同时在架构设计上大幅优化,目标是让...
- 比分网开发技术栈与功能详解_比分网有哪些
-
一、核心功能模块一个基本的比分网通常包含以下模块:首页/总览实时比分看板:滚动展示所有正在进行的比赛,包含比分、比赛时间、红黄牌等关键信息。热门赛事/焦点战:突出显示重要的、关注度高的比赛。赛事导航...
- 设计模式之-生成器_一键生成设计
-
一、【概念定义】——“分步构建复杂对象,隐藏创建细节”生成器模式(BuilderPattern):一种“分步构建型”创建型设计模式,它将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建...
- 构建第一个 Kotlin Android 应用_kotlin简介
-
第一步:安装AndroidStudio(推荐IDE)AndroidStudio是官方推荐的Android开发集成开发环境(IDE),内置对Kotlin的完整支持。1.下载And...
- 一周热门
-
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
飞牛OS入门安装遇到问题,如何解决?
-
系统C盘清理:微信PC端文件清理,扩大C盘可用空间步骤
-
10款高性能NAS丨双十一必看,轻松搞定虚拟机、Docker、软路由
-
python使用fitz模块提取pdf中的图片
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)