MySQL事务详解:从理论到实践,保障数据一致性

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 前言
  • 一、为什么需要事务?
  • 二、事务的四大特性(ACID)
  • 三、MySQL事务的使用方法
    • 1. 基本语法
    • 2. 自动提交的设置
    • 3. 保存点(Savepoint)
  • 四、事务隔离级别与并发问题
  • 五、MyBatis中的事务管理
    • 1. 编程式事务
    • 2. 声明式事务(整合Spring)
  • 六、事务的最佳实践
  • 总结


前言

在数据库操作中,事务(Transaction) 是确保数据一致性和完整性的核心机制。无论是转账、订单处理还是库存管理,事务都扮演着重要角色。本文将从基础概念出发,结合实例讲解MySQL事务的四大特性、使用方法及扩展知识,助你全面掌握事务的应用场景和最佳实践。


一、为什么需要事务?

想象一个转账场景:

  • 张三向李四转账100元
  • 王五向张三转账100元

对应的账户表结构如下:

id    money
1     100   -- 张三的账户  
2     300   -- 李四的账户  

若直接执行两条SQL:

UPDATE account SET money = 300 WHERE id = 1;  -- 张三账户+100  
UPDATE account SET money = 100 WHERE id = 2;  -- 李四账户-100  

问题:如果第二条SQL执行失败(例如拼写错误id=2w),张三的账户已变更,但李四的账户未扣款,导致数据不一致。

解决方案:通过事务将多个操作绑定为一个原子单元,要么全部成功,要么全部回滚。

二、事务的四大特性(ACID)

  1. 原子性(Atomicity)

    • 事务中的操作要么全部成功,要么全部失败。

    • 例如转账时,两条UPDATE必须同时成功或同时撤销。

  2. 一致性(Consistency)

    • 事务执行前后,数据库必须保持一致性状态。

    • 转账前后,总金额应保持不变(张三+100,李四-100)。

  3. 隔离性(Isolation)

    • 多个并发事务之间互不干扰。

    • 例如A事务修改数据时,B事务不能读取中间状态。

  4. 持久性(Durability)

    • 事务提交后,修改永久保存,即使系统故障也不丢失。

三、MySQL事务的使用方法

1. 基本语法

sql语句

START TRANSACTION;  -- 开启事务  
UPDATE account SET money = 300 WHERE id = 1;  
UPDATE account SET money = 100 WHERE id = 2;  
COMMIT;            -- 提交事务(持久化)  
-- 若发生错误:  
ROLLBACK;          -- 回滚事务(撤销所有操作)  

2. 自动提交的设置

MySQL默认开启自动提交(autocommit=1),每条SQL独立成事务。可通过以下命令修改:

SET autocommit = 0;  -- 关闭自动提交  

3. 保存点(Savepoint)

在复杂事务中设置回滚点:

SAVEPOINT sp1;  
-- 执行部分操作  
ROLLBACK TO sp1;  -- 回滚到sp1  

四、事务隔离级别与并发问题

MySQL支持四种隔离级别(默认为REPEATABLE READ):

  1. READ UNCOMMITTED

    • 问题:脏读(读到未提交的数据)。
  2. READ COMMITTED

    • 解决脏读,但存在不可重复读(同一查询结果不一致)。
  3. REPEATABLE READ

    • 解决不可重复读,但存在幻读(新增数据影响结果)。
  4. SERIALIZABLE

    • 完全串行化,解决所有问题,但性能最低。

设置隔离级别:

SET TRANSACTION ISOLATION LEVEL READ COMMITTED;  

五、MyBatis中的事务管理

在MyBatis中,事务可通过以下方式管理:

1. 编程式事务

SqlSession sqlSession = sqlSessionFactory.openSession(false);  // 关闭自动提交  
try {  userMapper.updateAccount1();  userMapper.updateAccount2();  sqlSession.commit();  
} catch (Exception e) {  sqlSession.rollback();  
}  

2. 声明式事务(整合Spring)

在Spring配置文件中声明事务管理器:

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">  <property name="dataSource" ref="dataSource"/>  
</bean>  
<tx:annotation-driven/>  

在Service层使用注解:

@Transactional(isolation = Isolation.DEFAULT, propagation = Propagation.REQUIRED)  
public void transfer() { ... }  

六、事务的最佳实践

  1. 尽量缩短事务时间

    • 避免长事务占用资源,引发锁竞争。
  2. 合理选择隔离级别

    • 根据业务需求平衡一致性与性能。
  3. 处理异常

    • 确保代码中捕获异常并回滚事务。
  4. 避免跨库事务

    • 分布式事务复杂度高,建议使用消息队列或最终一致性方案。

总结

事务是数据库系统的基石,通过ACID特性保障数据安全。合理使用事务能有效避免数据错乱,提升系统可靠性。在实际开发中,需结合业务场景选择隔离级别,并注意事务的粒度与性能优化。
记住:没有银弹,事务虽强大,滥用则成负担!


通过本文,你不仅掌握了事务的核心概念和MySQL操作方法,还了解了隔离级别、MyBatis集成及最佳实践。现在,尝试在你的项目中应用这些知识,感受事务带来的数据安全感吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/35680.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Next App Router(下)

五、loading 新增 app/loading.tsx 页面 const Loading () > {return <div>Loading...</div>; }; export default Loading;修改 app/page.tsx页面 /** 假设为一个获取数字的api */ const fetch_getNumber async (): Promise<number> > {return ne…

【JAVA】】深入浅出了解cookie、session、jwt

文章目录 前言一、首先了解http的cookie是什么&#xff1f;Cookie 属性及其含义1. NameValue2. Expires3. Max-Age4. Domain5. Path6. Secure7. HttpOnly8. SameSite示例 Cookie 分类1. Session Cookies2. Persistent Cookies3. First-Party Cookies4. Third-Party Cookies 二、…

【css酷炫效果】纯CSS实现粒子旋转动画

【css酷炫效果】纯CSS实现粒子旋转动画 缘创作背景html结构css样式完整代码效果图 想直接拿走的老板&#xff0c;链接放在这里&#xff1a;https://download.csdn.net/download/u011561335/90492008 缘 创作随缘&#xff0c;不定时更新。 创作背景 刚看到csdn出活动了&…

C++Lambda表达式

Lambda表达式 什么是Lambda表达式 ​ C11的颁布让C丰富了起来&#xff0c;任何一本介绍C11的书籍&#xff0c;都不可能跳过这一个点——Lambda表达式。人们经常称Lambda表达式是一个语法糖&#xff0c;说明这是一个”没有没事&#xff0c;有了更好“的一种语法表达&#xff0…

每天五分钟深度学习框架pytorch:基于pytorch搭建循环神经网络RNN

本文重点 我们前面介绍了循环神经网络RNN,主要分析了它的维度信息,其实它的维度信息是最重要的,一旦我们把维度弄清楚了,一起就很简单了,本文我们正式的来学习一下,如何使用pytorch搭建循环神经网络RNN。 RNN的搭建 在pytorch中我们使用nn.RNN()就可以创建出RNN神经网络…

el-table树形表格合并相同的值

el-table树形表格合并相同的值 el-table树形表格合并相同的值让Ai进行优化后的代码 el-table树形表格合并相同的值 <style lang"scss" scoped> .tableBox {/deep/ &.el-table th:first-child,/deep/ &.el-table td:first-child {padding-left: 0;} } …

2025年3月19日 十二生肖 今日运势

小运播报&#xff1a;2025年3月19日&#xff0c;星期三&#xff0c;农历二月二十 &#xff08;乙巳年己卯月丁亥日&#xff09;&#xff0c;法定工作日。 红榜生肖&#xff1a;兔、虎、羊 需要注意&#xff1a;猪、猴、蛇 喜神方位&#xff1a;正南方 财神方位&#xff1a;…

Git——分布式版本控制工具使用教程

本文主要介绍两种版本控制工具——SVN和Git的概念&#xff0c;接着会讲到Git的安装&#xff0c;Git常用的命令&#xff0c;以及怎么在Vscode中使用Git。帮助新手小白快速上手Git。 1. SVN和Git介绍 1.1 SVN 集中式版本控制工具&#xff0c;版本库是集中存放在中央服务器的&am…

QT5.15.2加载pdf为QGraphicsScene的背景

5.15.2使用pdf 必须要安装QT源码&#xff0c;可以看到编译器lib目录已经有pdf相关的lib文件&#xff0c;d是debug 1.找到源码目录&#xff1a;D:\soft\QT\5.15.2\Src\qtwebengine\include 复制这两个文件夹到编译器的包含目录中:D:\soft\QT\5.15.2\msvc2019_64\include 2.找…

【H2O2 | 软件开发】前端深拷贝的实现

目录 前言 开篇语 准备工作 正文 概述 JSON方法 递归 其他 结束语 前言 开篇语 本系列为短篇&#xff0c;每次讲述少量知识点&#xff0c;无需一次性灌输太多的新知识点。该主题文章主要是围绕前端、全栈开发相关面试常见问题撰写的&#xff0c;希望对诸位有所帮助。…

Docker - 切换源 (Linux / macOS)

文章目录 Linux 系统macOS 系统 Linux 系统 修改配置文件&#xff1a;/etc/docker/daemon.json "registry-mirrors": ["https://docker.mirrors.ustc.edu.cn","https://hub-mirror.c.163.com"]验证是否修改成功&#xff1a; docker info重启 …

hcia复习

一、网络设备 1、交换机&#xff1a;&#xff08;1&#xff09;提供MAC地址表&#xff0c;转发数据&#xff1b; &#xff08;2&#xff09;每个接口是一个独立的冲突域&#xff1b; &#xff08;3&#xff09;凡是连在交换机上的所有设备都处于同一广播域&#xff08;网络&am…

opencv初步学习——图像处理3

这一部分我们将学习opencv中对图像大小进行调整的基本操作&#xff0c;以及掩模操作&#xff0c;我们直接进入正言 一、cv2.resize( )函数 1-1、组成与构造 该函数的作用就算用来帮助我们实现对图像大小的处理&#xff0c;具体的组成与构造如下&#xff1a; cv2.resize(src , …

[LevelDB]关于LevelDB存储架构到底怎么设计的?

本文内容组织形式 LevelDB 存储架构重要特点总体概括LevelDB中内存模型MemTableMemTable的数据结构背景&#xff1a;SkipListSkiplist的数据结构 Skiplist的数据访问细节 SkipList的核心方法Node细节源代码 MemTable的数据加速方式Iterator 的核心方法 MemTable 的读取&写入…

【存储中间件】Redis核心技术与实战(四):Redis高并发高可用(Redis集群 Smart客户端、集群原理)

文章目录 Redis集群Smart客户端smart客户端原理ASK 重定向集群下的Jedis客户端Hash tags 集群原理节点通信通信流程Gossip 消息节点选择 故障转移故障发现主观下线客观下线 故障恢复资格检查准备选举时间发起选举选举投票替换主节点 故障转移时间 集群不可用判定集群读写分离 个…

【接口耗时】⭐️自定义拦截器实现接口耗时统计

&#x1f4a5;&#x1f4a5;✈️✈️欢迎阅读本文章❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;本篇文章阅读大约耗时三分钟。 ⛳️motto&#xff1a;不积跬步、无以千里 &#x1f4cb;&#x1f4cb;&#x1f4cb;本文目录如下&#xff1a;&#x1f381;&#x1f381;&a…

杨校老师课堂之编程入门与软件安装【图文笔记】

亲爱的同学们&#xff0c;热烈欢迎踏入青少年编程的奇妙世界&#xff01; 我是你们的授课老师杨校 &#xff0c;期待与大家一同开启编程之旅。 1. 轻松叩开编程之门 1.1 程序的定义及生活中的应用 程序是人与计算机沟通的工具。在日常生活中&#xff0c;像手机里的各类 APP、电…

【从零开始】Air780EPM的LuatOS二次开发——OneWire协议调试注意事项!

当涉及到与传感器、执行器等外部设备交互时&#xff0c;OneWire协议的高效调试成为决定项目成败的关键环节。OneWire协议&#xff08;单总线协议&#xff09;以其仅需一根数据线即可实现设备通信的极简特性&#xff0c;被广泛应用于温度传感器、身份识别模块等场景。 一、LuatO…

redis数据结构、多路复用、持久化---java

数据结构 Redis 提供了丰富的数据类型&#xff0c;常见的有五种数据类型&#xff1a;String&#xff08;字符串&#xff09;&#xff0c;Hash&#xff08;哈希&#xff09;&#xff0c;List&#xff08;列表&#xff09;&#xff0c;Set&#xff08;集合&#xff09;、Zset&am…

vue3之写一个aichat ----vite.config.js

vite.config.js的CSS配置 postcss-pxtorem 开发响应式网页的时候需要用到postcss-pxtorem amfe-flexible amfe-flexible是由阿里团队开发的一个库&#xff0c;它可以根据设备的屏幕宽度去动态调整HTML根元素()的字体大小&#xff0c;这意味着无论用户使用什么尺寸的设备访问你…