当前位置: 首页 > news >正文

Python 操作 MySQL 事务:从入门到避坑

在实际开发中,单条 SQL 往往不够用。转账、订单处理、库存扣减……这些场景要求多条 SQL要么全成功,要么全失败。这就是事务存在的意义。

本文用 Python 实战讲解如何正确使用 MySQL 事务,覆盖pymysqlmysql-connector-pythonSQLAlchemy三种主流方式。


一、先搞懂事务的核心:ACID

特性含义举例
Atomicity(原子性)操作不可分割,全做或全不做转账:扣款和入账必须同时成功
Consistency(一致性)事务前后数据保持一致余额不能凭空消失
Isolation(隔离性)并发事务互不干扰两人同时取钱,不能互相影响
Durability(持久性)提交后数据永久保存服务器重启数据不丢失

一句话:事务就是保证数据不出乱子的机制。


二、方式一:pymysql(最常用)

2.1 基本用法

importpymysql conn=pymysql.connect(host='localhost',user='root',password='your_password',database='test_db',charset='utf8mb4')try:withconn.cursor()ascursor:# 开启事务(默认就是手动提交模式)sql1="UPDATE accounts SET balance = balance - 100 WHERE user_id = 1"sql2="UPDATE accounts SET balance = balance + 100 WHERE user_id = 2"cursor.execute(sql1)cursor.execute(sql2)# 全部成功,提交conn.commit()print("转账成功")exceptExceptionase:# 任何一步出错,回滚conn.rollback()print(f"转账失败,已回滚:{e}")finally:conn.close()

2.2 关键点

  • conn.commit()提交事务
  • conn.rollback()回滚事务
  • 出错必须回滚,否则已执行的 SQL 不会撤销
  • 默认autocommit=False,所以需要手动提交

三、方式二:mysql-connector-python(官方驱动)

importmysql.connector conn=mysql.connector.connect(host='localhost',user='root',password='your_password',database='test_db')cursor=conn.cursor()try:cursor.execute("UPDATE accounts SET balance = balance - 100 WHERE user_id = 1")cursor.execute("UPDATE accounts SET balance = balance + 100 WHERE user_id = 2")conn.commit()exceptmysql.connector.Erroraserr:conn.rollback()print(f"Error:{err}")finally:cursor.close()conn.close()

pymysql逻辑一致,只是 API 略有不同。


四、方式三:SQLAlchemy ORM(推荐大型项目)

fromsqlalchemyimportcreate_engine,Column,Integer,Stringfromsqlalchemy.ormimportsessionmaker,declarative_base Base=declarative_base()classAccount(Base):__tablename__='accounts'id=Column(Integer,primary_key=True)name=Column(String(50))balance=Column(Integer)engine=create_engine('mysql+pymysql://root:password@localhost/test_db')Session=sessionmaker(bind=engine)session=Session()try:account1=session.query(Account).filter_by(id=1).with_for_update().first()account2=session.query(Account).filter_by(id=2).with_for_update().first()account1.balance-=100account2.balance+=100session.commit()print("转账成功")exceptExceptionase:session.rollback()print(f"转账失败:{e}")finally:session.close()

为什么用with_for_update()

普通查询在并发下可能读到脏数据。with_for_update()加行锁,确保这条记录在事务结束前不被其他事务修改,解决并发问题。


五、三种方式对比

维度pymysqlmysql-connectorSQLAlchemy
上手难度⭐⭐⭐⭐⭐⭐⭐
性能稍慢(有 ORM 开销)
适用场景轻量脚本、小项目官方驱动、稳定需求中大型项目
并发控制手动写 SQL手动写 SQLwith_for_update()内置

选型建议:小项目用pymysql,追求稳定用官方驱动,项目大了直接上 SQLAlchemy。


六、常见坑 & 最佳实践

坑1:异常没捕获,事务没回滚

# ❌ 错误示范cursor.execute(sql1)cursor.execute(sql2)# 如果这里报错,sql1 已执行但没回滚conn.commit()

必须用 try/except 包裹,except 里调用rollback()

坑2:连接池里的事务混乱

用连接池时,确保一个连接只处理一个事务,不要跨连接做事务操作。

坑3:忘了设置隔离级别

MySQL 默认隔离级别是REPEATABLE READ,但有些场景需要READ COMMITTED

conn.begin()# 显式开启事务cursor.execute("SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED")

最佳实践清单

  • ✅ 事务里的 SQL 尽量少,减少锁持有时间
  • ✅ 捕获所有异常,确保回滚
  • ✅ 高并发场景加行锁(SELECT ... FOR UPDATE
  • ✅ 生产环境用连接池(如DBUtilsSQLAlchemy内置池)
  • ✅ 不要在事务里做网络请求、文件 IO 等耗时操作

七、总结

你的场景推荐方案
写个脚本批量处理数据pymysql+ 手动 commit/rollback
官方项目,求稳定mysql-connector-python
Web 项目、多人协作SQLAlchemy+with_for_update()

事务不复杂,但用错了比不用更危险。记住三个动作:begin → commit / rollback → close,就能覆盖 90% 的场景。

http://www.cnnetsun.cn/news/2660620.html

相关文章:

  • 别只盯着平均响应时间!用JMeter汇总报告做性能对比分析的3个实战技巧
  • 共识机制:当三个 Agent 意见不一致时,系统该听谁的?
  • Gemini报告里的异常信号你真的看懂了吗?资深AI架构师教你用3层归因法锁定根因
  • 2026视频提取字幕保姆级教程:制作方法+工具推荐手把手教你
  • Motrix浏览器插件:告别龟速下载,体验终极加速方案
  • Live Room Watcher:直播间数据流架构深度解析与实时监控技术实现
  • 嵌入式Linux电源管理实战:GPIO驱动中的pm_runtime_get_sync到底在做什么?以Zynq平台为例
  • OxyPlot高性能跨平台绘图库:.NET数据可视化深度集成与架构解析
  • 不只是打孔:用Allegro 17.4 Via Array 功能,5分钟搞定PCB板边与电源铺铜的过孔阵列
  • 微软商店装WSL2太占C盘?试试这个‘先装后移’的野路子(Ubuntu 20.04实测)
  • Zotero终极美化插件:打造专业高效的文献管理界面
  • TimeMixer深度解析:如何通过全MLP架构实现多尺度时间序列预测的5大优势
  • 基于Arduino与无源蜂鸣器的电子钢琴制作:从硬件搭建到软件编程全解析
  • 基于ESP32-CAM与YOLO的自主格斗机器人:低成本嵌入式AI实践
  • 科技行业性别平等:从权力结构到系统变革的破局之路
  • Excel高手私藏技巧:用XLOOKUP函数实现动态下拉菜单与数据联动(附模板)
  • ARM DynamIQ架构下Stash操作与缓存一致性处理
  • 英雄联盟玩家必备:League Akari 本地化智能助手完整指南
  • VOFA+上位机连接ESP32:三种协议(FireWater/JustFloat)实战性能对比与避坑指南
  • 实战复盘:用Python+Requests搞定WIPO专利站那个烦人的六宫格验证码(附完整代码)
  • Windows 服务全攻略:从命令行创建到自动化运维的艺术
  • 实时BPM分析器终极指南:三分钟掌握音频节拍检测核心技术
  • 免费开源工具Ofd2Pdf:3分钟实现OFD转PDF的终极解决方案
  • 告别CLI翻译思维:从Juniper模型看如何用YANG设计出清晰好用的网络数据模型
  • 保姆级教程:用MATLAB的Hyperspectral Imaging Library搞定高光谱图像RGB可视化
  • 基于Arduino与BioAmp传感器的心电信号采集与可视化系统搭建指南
  • 从战斗机到家用车:聊聊HUD技术的前世今生与未来AR导航怎么玩
  • B站视频格式转换完整教程:让缓存视频重获新生的终极指南
  • 为什么92%的Gemini集群在QPS破万后出现隐性OOM?深度拆解内存隔离、CUDA上下文缓存与cgroup v2的致命协同失效
  • Windows系统终极管理工具:WinUtil一键优化完整指南