Python 使用 MySQL 数据库进行事务处理步骤

文章目录

一、事务核心概念(先理解再实操)

事务(Transaction)是数据库操作的最小逻辑单元,遵循 ACID 原则

  • 原子性(Atomicity):要么全部执行成功,要么全部失败回滚
  • 一致性(Consistency):执行前后数据库状态保持一致
  • 隔离性(Isolation):多个事务互不干扰
  • 持久性(Durability):提交后数据永久保存

Python操作MySQL时,默认是自动提交(autocommit) 模式(执行单条SQL会立即生效),而事务处理需要先关闭自动提交,手动控制提交/回滚。

二、完整事务处理代码示例

以「转账场景」为例(经典的事务应用场景:A账户扣钱、B账户加钱,必须同时成功/失败):

python 复制代码
import mysql.connector
from mysql.connector import Error

def transfer_money(from_id, to_id, amount):
    """
    模拟转账的事务处理
    :param from_id: 转出账户ID
    :param to_id: 转入账户ID
    :param amount: 转账金额
    """
    connection = None
    try:
        # 1. 建立数据库连接
        connection = mysql.connector.connect(
            host='localhost',
            user='root',
            password='123456',
            database='test_db'
        )

        # 2. 关闭自动提交,开启手动事务控制
        connection.autocommit = False

        cursor = connection.cursor()

        # 3. 执行事务内的多个SQL操作
        # 步骤1:扣除转出账户金额
        deduct_sql = "UPDATE account SET balance = balance - %s WHERE id = %s"
        cursor.execute(deduct_sql, (amount, from_id))
        
        # 步骤2:增加转入账户金额
        add_sql = "UPDATE account SET balance = balance + %s WHERE id = %s"
        cursor.execute(add_sql, (amount, to_id))

        # 模拟异常(可取消注释测试回滚效果)
        # raise Error("模拟转账异常,触发回滚")

        # 4. 所有操作执行成功,提交事务
        connection.commit()
        print("转账成功!事务已提交")

    except Error as e:
        # 5. 发生异常,回滚事务(撤销所有已执行的SQL操作)
        if connection:
            connection.rollback()
        print(f"转账失败,事务已回滚!错误信息:{e}")

    finally:
        # 6. 释放资源(关闭游标和连接)
        if connection and connection.is_connected():
            cursor.close()
            # 恢复自动提交(可选,不影响,但规范)
            connection.autocommit = True
            connection.close()
            print("数据库连接已关闭")

# ==================== 测试前准备 ====================
# 先在MySQL中创建测试表和数据:
# CREATE DATABASE IF NOT EXISTS test_db;
# USE test_db;
# CREATE TABLE IF NOT EXISTS account (
#     id INT PRIMARY KEY,
#     name VARCHAR(50),
#     balance DECIMAL(10,2)
# );
# INSERT INTO account (id, name, balance) VALUES (1, '张三', 1000.00), (2, '李四', 500.00);

# ==================== 执行转账测试 ====================
# 测试正常转账(张三给李四转200元)
transfer_money(from_id=1, to_id=2, amount=200.00)

三、关键操作解释

  1. 关闭自动提交
    connection.autocommit = False 是开启事务的核心,关闭后执行的SQL不会立即生效,需手动 commit()
  2. 事务提交
    connection.commit() 仅当所有SQL执行无异常时调用,会将事务内所有操作永久写入数据库。
  3. 事务回滚
    connection.rollback() 仅在捕获到异常时调用,会撤销事务内所有已执行的SQL操作,恢复到事务开始前的状态。
  4. 异常捕获
    必须用 try-except 包裹所有事务内的操作,确保任何错误都能触发回滚,避免数据不一致。

四、拓展场景:批量操作事务

如果需要批量插入/更新多条数据,事务同样适用,示例如下:

python 复制代码
def batch_insert_users(users):
    """批量插入用户,使用事务保证全部成功/失败"""
    connection = None
    try:
        connection = mysql.connector.connect(
            host='localhost',
            user='root',
            password='123456',
            database='test_db'
        )
        connection.autocommit = False
        cursor = connection.cursor()

        insert_sql = "INSERT INTO user (name, age) VALUES (%s, %s)"
        # 批量执行SQL(效率更高)
        cursor.executemany(insert_sql, users)

        connection.commit()
        print(f"批量插入 {cursor.rowcount} 条数据成功")

    except Error as e:
        if connection:
            connection.rollback()
        print(f"批量插入失败,事务回滚:{e}")
    finally:
        if connection and connection.is_connected():
            cursor.close()
            connection.autocommit = True
            connection.close()

# 测试批量插入
user_list = [("王五", 30), ("赵六", 28), ("孙七", 35)]
batch_insert_users(user_list)
相关推荐
小oo呆3 分钟前
【学习心得】Python的Pydantic(简介)
前端·javascript·python
岚天start4 分钟前
【日志监控方案】Python脚本获取关键字日志信息并推送钉钉告警
python·钉钉·日志监控
tyatyatya5 分钟前
MySQL Group Replication(MGR)集群部署,实现自动故障切换
数据库·mysql
b***59435 分钟前
mysql 迁移达梦数据库出现的 sql 语法问题 以及迁移方案
数据库·sql·mysql
木风小助理6 分钟前
MySQL中COUNT()、COUNT(1)与COUNT
数据库
叫我:松哥6 分钟前
基于 Flask 框架开发的在线学习平台,集成人工智能技术,提供分类练习、随机练习、智能推荐等多种学习模式
人工智能·后端·python·学习·信息可视化·flask·推荐算法
rgeshfgreh6 分钟前
Python环境管理:uv极速对决Conda全能
python
不想上班的小吕6 分钟前
采购申请创建(BAPI_PR_CREATE/BAPI_REQUISITION_CREATE)
java·服务器·数据库
幻云20107 分钟前
Python机器学习:从入门到精通
python
j***89468 分钟前
MySQL官网驱动下载(jar包驱动和ODBC驱动)【详细教程】
数据库·mysql