宋跃江 發表於 2026-1-8 11:16:13

Python 使用 MySQL 数据库进行事务处理完整示例

<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li><a href="#_label0">一、事务核心概念(先理解再实操)</a></li><li><a href="#_label1">二、完整事务处理代码示例</a></li><li><a href="#_label2">三、关键操作解释</a></li><li><a href="#_label3">四、拓展场景:批量操作事务</a></li></ul></div><p class="maodian"><a name="_label0"></a></p><h2>一、事务核心概念(先理解再实操)</h2>
<p>事务(Transaction)是数据库操作的最小逻辑单元,遵循 <strong>ACID 原则</strong>:</p>
<ul><li><strong>原子性(Atomicity)</strong>:要么全部执行成功,要么全部失败回滚</li><li><strong>一致性(Consistency)</strong>:执行前后数据库状态保持一致</li><li><strong>隔离性(Isolation)</strong>:多个事务互不干扰</li><li><strong>持久性(Durability)</strong>:提交后数据永久保存</li></ul>
<p>Python操作MySQL时,默认是<strong>自动提交(autocommit)</strong> 模式(执行单条SQL会立即生效),而事务处理需要先关闭自动提交,手动控制提交/回滚。</p>
<p class="maodian"><a name="_label1"></a></p><h2>二、完整事务处理代码示例</h2>
<p>以「转账场景」为例(经典的事务应用场景:A账户扣钱、B账户加钱,必须同时成功/失败):</p>
<div class="jb51code"><pre class="brush:py;">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)</pre></div>
<p class="maodian"><a name="_label2"></a></p><h2>三、关键操作解释</h2>
<ol><li><strong>关闭自动提交</strong>:<br /><code>connection.autocommit = False</code> 是开启事务的核心,关闭后执行的SQL不会立即生效,需手动 <code>commit()</code>。</li><li><strong>事务提交</strong>:<br /><code>connection.commit()</code> 仅当所有SQL执行无异常时调用,会将事务内所有操作永久写入数据库。</li><li><strong>事务回滚</strong>:<br /><code>connection.rollback()</code> 仅在捕获到异常时调用,会撤销事务内所有已执行的SQL操作,恢复到事务开始前的状态。</li><li><strong>异常捕获</strong>:<br />必须用 <code>try-except</code> 包裹所有事务内的操作,确保任何错误都能触发回滚,避免数据不一致。</li></ol>
<p class="maodian"><a name="_label3"></a></p><h2>四、拓展场景:批量操作事务</h2>
<p>如果需要批量插入/更新多条数据,事务同样适用,示例如下:</p>
<div class="jb51code"><pre class="brush:py;">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)</pre></div>
頁: [1]
查看完整版本: Python 使用 MySQL 数据库进行事务处理完整示例