[Django学习]如何在Django中声明“事务”(包括事务回滚)

一、事务概念

事务(transaction)是访问并可能操作各种数据项的一个数据库操作序列,这些操作要么全部执行,要么全部不执行,是一个不可分割的工作单位。比如某软件的转账功能:现在用户A要给B转账300块,其流程为,先检查A的余额,若余额大于300,则将A的余额减去300,之后将B的余额增加300。

在上述例子中,A余额减300,和B余额增加300这两个操作要么都执行,要么一个也不执行,否则可能会出现A的余额减300后,B的余额没有变化,或是B的余额凭空增加300,但A的余额还是不变。转账这一系列操作为一个事务。

但若一开始发现A的余额不足300块,则对事务进行回滚,取消之后的一系列操作。

二、Django声明事务的两种方法

1.装饰器方法

使用@transaction.atomic装饰器来装饰一个视图函数或方法。在这个函数或方法内部的所有数据库操作都将在一个事务中执行。如下面这个代码:

python 复制代码
from django.db import transaction  
from .models import Account  
  
@transaction.atomic  
def transfer_funds(from_account_id, to_account_id, amount):  
    try:  
        # 从A账户中减去金额  
        from_account = Account.objects.get(pk=from_account_id)  
        if from_account.balance < amount:  # 余额不足,回滚
            raise ValueError("账户余额不足")  
        from_account.balance -= amount  
        from_account.save()  
  
        # 向账户B中增加金额  
        to_account = Account.objects.get(pk=to_account_id)  
        to_account.balance += amount  
        to_account.save()  
  
    except Exception as e:  
        # 如果在此过程中发生任何异常,Django会自动回滚事务  
        print(f"An error occurred: {e}")    

    # 代码块成功执行完毕,Django会自动提交事务

2.上下文管理器:

也可以不把整个视图作为事务处理,而是在一个代码块里使用transaction.atomic()上下文管理器。

python 复制代码
from django.db import transaction  
from .models import Account  
  
def transfer_funds(from_account_id, to_account_id, amount):  
    try:  
        with transaction.atomic():  
            #  这里放刚刚转账操作的代码  
    except Exception as e:  
        print(f"操作发生错误: {e}")
相关推荐
一行玩python20 分钟前
SQLAlchemy,ORM的Python标杆!
开发语言·数据库·python·oracle
MXsoft61839 分钟前
华为服务器(iBMC)硬件监控指标解读
大数据·运维·数据库
颜淡慕潇40 分钟前
【K8S系列】kubectl describe pod显示ImagePullBackOff,如何进一步排查?
后端·云原生·容器·kubernetes
TheITSea1 小时前
云服务器宝塔安装静态网页 WordPress、VuePress流程记录
java·服务器·数据库
Clarify1 小时前
docker部署go游戏服务器(进阶版)
后端
IT书架2 小时前
golang面试题
开发语言·后端·golang
王ASC2 小时前
ORA-01461: 仅能绑定要插入 LONG 列的 LONG 值。ojdbc8版本23.2.0.0驱动BUG【已解决】
数据库·sql·oracle
Dlwyz2 小时前
问题: redis-高并发场景下如何保证缓存数据与数据库的最终一致性
数据库·redis·缓存
机器之心3 小时前
全球十亿级轨迹点驱动,首个轨迹基础大模型来了
人工智能·后端
如意机反光镜裸3 小时前
如何快速将Excel数据导入到SQL Server数据库
数据库