Python使用MySQL事务的三种主流方式

 更新时间:2026年05月31日 08:24:52   作者:detayun  
本文详细讲解了如何在Python中使用MySQL事务,覆盖了pymysql、mysql-connector-python和SQLAlchemy三种主流方式,重点介绍了事务的ACID特性、常见用法及最佳实践,需要的朋友可以参考下

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

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

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

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

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

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

2.1 基本用法

import pymysql

conn = pymysql.connect(
    host='localhost',
    user='root',
    password='your_password',
    database='test_db',
    charset='utf8mb4'
)

try:
    with conn.cursor() as cursor:
        # 开启事务(默认就是手动提交模式)
        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("转账成功")
except Exception as e:
    # 任何一步出错,回滚
    conn.rollback()
    print(f"转账失败,已回滚:{e}")
finally:
    conn.close()

2.2 关键点

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

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

import mysql.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()
except mysql.connector.Error as err:
    conn.rollback()
    print(f"Error: {err}")
finally:
    cursor.close()
    conn.close()

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

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

from sqlalchemy import create_engine, Column, Integer, String
from sqlalchemy.orm import sessionmaker, declarative_base

Base = declarative_base()

class Account(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 -= 100
    account2.balance += 100
    
    session.commit()
    print("转账成功")
except Exception as e:
    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% 的场景。

以上就是Python使用MySQL事务的三种主流方式的详细内容,更多关于Python使用MySQL事务的资料请关注脚本之家其它相关文章!

相关文章

  • 一文详解Python中的super 函数

    一文详解Python中的super 函数

    这篇文章主要介绍了一文了解Python中的super 函数,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下
    2022-09-09
  • Python切换pip安装源的方法详解

    Python切换pip安装源的方法详解

    众所周知pip是Python中非常方便易用的安装包管理器,但是在实际安装中,却是非常的慢,该如何解决呢?那么下面这篇文章就给大家介绍了Python切换pip安装源的方法,文中介绍的很详细,对大家学习或者理解具有一定的参考借鉴价值,有需要的朋友们下面来一起看看吧。
    2016-11-11
  • Anaconda下安装mysql-python的包实例

    Anaconda下安装mysql-python的包实例

    今天小编就为大家分享一篇Anaconda下安装mysql-python的包实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-06-06
  • 在Python中操作字典之setdefault()方法的使用

    在Python中操作字典之setdefault()方法的使用

    这篇文章主要介绍了在Python中操作字典之setdefault()方法的使用,是Python入门学习中的基础知识,需要的朋友可以参考下
    2015-05-05
  • 浅谈Python中的私有变量

    浅谈Python中的私有变量

    本篇文章给大家讲述了Python中的私有变量相关的知识点内容,学习后大家会对私有变量有更系统的了解。
    2018-02-02
  • OpenCV实现车辆识别和运动目标检测

    OpenCV实现车辆识别和运动目标检测

    本文主要介绍了OpenCV实现车辆识别和运动目标检测,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • Python使用asyncio处理异步编程的代码示例

    Python使用asyncio处理异步编程的代码示例

    在 Python 中,异步编程可以使用 asyncio 库,该库提供了一些工具和功能来编写异步代码,本文介绍了处理异步编程的几个关键概念和示例,需要的朋友可以参考下
    2024-07-07
  • Python基于network模块制作电影人物关系图

    Python基于network模块制作电影人物关系图

    这篇文章主要介绍了Python基于network模块制作电影人物关系图,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-06-06
  • 详解Python 中sys.stdin.readline()的用法

    详解Python 中sys.stdin.readline()的用法

    这篇文章主要介绍了Python 中sys.stdin.readline()的用法,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-09-09
  • 浅谈Selenium 控制浏览器的常用方法

    浅谈Selenium 控制浏览器的常用方法

    这篇文章主要介绍了浅谈Selenium 控制浏览器的常用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12

最新评论