我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Django数据库--事务及事务回滚

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Django数据库--事务及事务回滚

数据库的读写操作中,事务在保证数据的安全性和一致性方面起着关键的作用,而回滚正是这里面的核心操作。Django的ORM在事务方面也提供了不少的API。有事务出错的整体回滚操作,也有基于保存点的部分回滚。本文将讨论Django中的这两种机制的运行原理。

 

Django利用django.db.transaction模块中的API对数据库进行事务的管理

Django provides a straightforward API in the django.db.transaction module to manage the autocommit state of each database connection.

 

主要函数:

1. get_autocommit(using=None)   

判断事务是否自动提交

2. set_autocommit(autocommit, using=None)  

设置自动提交事务

 

这些函数使接受一个 using 参数表示所要操作的数据库。如果未提供,则 Django 使用 "default" 数据库。

 

3. on_commit(do something)

事务提交后马上执行任务,例如celery任务

例如:

with transation.atomic:

    #do something and commit the transaction

    transaction.on_commit(lambda: some_celery_task.delay('arg1'))

 

 

怎么使用?在哪里使用?

事务是一系列的数据库操作,在数据的安全性和减少网络请求方面都有很大的优势。关于数据库事务的文章有很多,我这里就不展开讨论了。

那么ORM中有哪些相关的API呢?

trasation模块中最重要的是一个Atomic类,Atomic是一个上下文管理器。可以使用@transaction.atomic 或者with transaction.atomic 的方式来调用。

 

为了设置保存点,即断点进行事务的执行和回滚,可以嵌套使用with transaction.atomic,例如官网的例子(伪代码):

with transaction.atomic():       # Outer atomic, start a new transaction
    transaction.on_commit(foo)      #事务提交后马上执行foo函数

    try:
        with transaction.atomic():      # Inner atomic block, create a savepoint
            transaction.on_commit(bar)      #事务提交后马上执行foo函数
            raise SomeError()      # Raising an exception - abort the savepoint
    except SomeError:
          pass

第一个with transaction.atomic()创建事务,第二个with transaction.atomic()创建保存点。

虽然错误raiseSomeError是从‘内部’的保存点发出来的,但只会影响到‘外部’的保存点,即只会回滚前面的数据库操作。

 

下面还会讨论另一种创建保存点的方法。

在使用transaction.atomic前需要注意的问题:

1. 数据库的自动提交默认为开启,如果要将它关闭,必须很小心。一旦使用了transaction,即关闭了自动提交。

2. 如果数据库之前的使用的是自动提交,那么在切换为非自动提交之前,必须确保当前没有活动的事务,通常可以手动执行commit() 或者 rollback() 函数来把未提交的事务提交或者回滚。

 

 

一、整体回滚

所有的数据库更新操作都会在一个事务中执行,如果事务中任何一个环节出现错误,都会回滚整个事务。

 

案例(伪代码1):

from django.db import transaction

# open a transaction
@transaction.atomic                #装饰器格式
def func_views(request):
         do_something()    
         a = A()              #实例化数据库模型
         try:
            a.save()
         except DatabaseError:
            pass

此方案整个view都会在事务之中,所有对数据库的操作都是原子性的。

 

案例(伪代码2):

from django.db import transaction

 
def func_views(request):
    try:
        with transaction.atomic():      #上下文格式,可以在python代码的任何位置使用
            a = A()
            a.save()
            #raise DatabaseError     #测试用,检测是否能捕捉错误
    except DatabaseError:     # 自动回滚,不需要任何操作
            pass

此方案比较灵活,事务可以在代码中的任意地方开启,对于事务开启前的数据库操作是必定会执行的,事务开启后的数据库操作一旦出现错误就会回滚。

 

需要注意的是:

1. python代码中对Models的修改和对数据库的修改的区别,数据库层面的修改不会影响Models实例变量。

如果在代码中修改一个变量,例如:

try:
        with transaction.atomic():     
            a = A()
            a.attribute = True   #A表的某一个属性(即数据库的某一列)
            a.save()
            raise  DatabaseError    
except DatabaseError:   
           pass

print(a.attribute)

#输出结果:True

 

即使数据库回滚了,但是a实例的变量a.attribute还是会保存在Models实例中,如果需要修改,就需要在except DatabaseError后面进行。

 

2. transaction不需要在代码中手动commit和rollback的。因为只有当一个transaction正常退出的时候,才会对数据库层面进行操作。除非我们手动调用transaction.commit和transaction.rollback

 

实际案例(此实例用伪代码2的格式):

models.py

数据表

class Author(models.Model):
    name = models.CharField(max_length=30,null=False)
    age = models.IntegerField()
    email = models.URLField(null=True)

class Count(models.Model):
    name = models.CharField(max_length=30)
    article_amount = models.IntegerField()

 

views.py

from django.shortcuts import render
from django.http import HttpResponse
from index.models import Author,Count
from django.db import transaction,IntegrityError

def add_author_views(request):
    author_name = u'renyingying'
    author = Author(name=author_name, age=24, email='renyingying@qqq.com')
    # author.save()

    count = Count(name=author_name, article_amount=1)
    count.save()

    try:
        with transaction.atomic():
            author.save()
            raise DatabaseError    #报出错误,检测事务是否能捕捉错误
    except DatabaseError:     # 自动回滚,不需要任何操作
            pass

事务外的数据库操作正常执行,而事务内的数据库操作则会回滚。

author表

 

 

count表

 

 

将raise DatabaseError这一行代码注释掉,author才会有数据

 

 

 

二、保存点Savepoint(断点回滚)

保存点是事务中的标记,从原理实现上来说是一个类似存储结构的类。可以回滚部分事务,而不是完整事务,同时会保存部分事务。python后端程序可以使用保存点。

一旦打开事务atomic(),就会构建一系列等待提交或回滚的数据库操作。通常,如果发出回滚命令,则会回滚整个事务。保存点则提供了执行细粒度回滚的功能,而不是将执行的完全回滚transaction.rollback()。

 

工作原理:savepoint通过对返回sid后面的将要执行的数据库操作进行计数,并保存在内置的列表中,当对数据库数据库进行操作时遇到错误而中断,根据sid寻找之前的保存点并回滚数据,并将这个操作从列表中删除。

 

相关API:

1. savepoint(using = None)

创建一个新的保存点。这表示处于正常状态的事务的一个点。返回保存点ID(sid)。在一个事务中可以创建多个保存点。

2. savepoint_commit(sid,using = None)

发布保存点sid,从创建保存点开始执行的数据库操作将成为可能回滚事务的一部分

3. savepoint_rollback(sid,using = None)

将事务回滚到保存点sid

4. clean_savepoints(using = None)

重置用于生成唯一保存点ID的计数器

值得注意的是:

这些函数中的每一个都接受一个using参数,该参数是数据库的名称。如果using未提供参数,则使用"default"默认数据库。

 

 

案例:

models.py上文的案例一样

 

views.py

from django.db import transaction


# open a transaction
@transaction.atomic
def add_author_views(request):
    # 自动提交方式
    # Author.objects.create(name=u'wangbaoqiang',age=33,email='wangbaoqiang@qqq.com')

    author_name = u'linghuchong'
    author = Author(name=author_name,age=26,email='linghuchong@qqq.com')
    author.save()
    # transaction now contains author.save()

    sid = transaction.savepoint()

    try:
        count = Count(name=author_name, article_amount=1)
        count.save()
        # transaction now contains author.save() and count.save()
        transaction.savepoint_commit(sid)
        # open transaction still contains author.save() and count.save()
    except IntegrityError:
        transaction.savepoint_rollback(sid)
        # open transaction now contains only count.save()
        # 保存author操作回滚后,事务只剩下一个操作 

   transaction.clean_savepoints()  #清除保存点

 

注意:希望当遇到错误得到回滚的事务一定要放在try里面(如果放在try外面,虽然不会报错,但是是不会执行的)。如上面的例子,如果在给Count表执行插入数据发生错误,就会‘断点’回滚到Count表插入数据前,Author表插入的数据不变。

 

结果显示:

Author表

 

Count表

 

参考文章:

https://blog.csdn.net/m0_37422289/article/details/82221489

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Django数据库--事务及事务回滚

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Django数据库--事务及事务回滚

数据库的读写操作中,事务在保证数据的安全性和一致性方面起着关键的作用,而回滚正是这里面的核心操作。Django的ORM在事务方面也提供了不少的API。有事务出错的整体回滚操作,也有基于保存点的部分回滚。本文将讨论Django中的这两种机制的
2023-01-30

Django事务回滚如何实现

这篇文章主要介绍“Django事务回滚如何实现”,在日常操作中,相信很多人在Django事务回滚如何实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Django事务回滚如何实现”的疑惑有所帮助!接下来,请跟
2023-07-05

MyBatis ORM的数据库事务回滚策略

MyBatis ORM 是一个优秀的持久层框架,它支持定制化 SQL、存储过程以及高级映射。MyBatis 遵循了 Java 的 JDBC 规范,因此在处理数据库事务时,也需要遵循 JDBC 的事务管理机制。MyBatis 的数据库事务回
MyBatis ORM的数据库事务回滚策略
2024-09-16

Django 事务回滚的具体实现

本文主要介绍了Django 事务回滚的具体实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-02-20

spring事务与数据库事务

1、事务的ACID属性 A(Atomicity):原子性,事务要么都成功,要么都失败,是一个整体 C(Consistency):一致性,事务前后数据保持一致,如张三去银行取钱,取了500,则张三到手500,银行扣除500,总数不变 I(Isolation):隔
spring事务与数据库事务
2015-03-25

SpringBoot数据层测试事务回滚如何实现

本文小编为大家详细介绍“SpringBoot数据层测试事务回滚如何实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“SpringBoot数据层测试事务回滚如何实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。数
2023-07-04
2024-04-02

Gorm 事务错误 error = 事务已提交或回滚

php小编柚子在这篇文章中将为大家介绍一个常见的错误信息:“Gorm 事务错误 error = 事务已提交或回滚”。在使用Gorm进行数据库操作时,有时候会遇到这个错误,让人困惑不解。本文将详细解释这个错误的原因以及可能的解决方案,帮助读者
Gorm 事务错误 error = 事务已提交或回滚
2024-02-11

怎么在golang中实现mysql数据库事务的提交与回滚

这期内容当中小编将会给大家带来有关怎么在golang中实现mysql数据库事务的提交与回滚,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。golang适合做什么golang可以做服务器端开发,但golang
2023-06-14

MySQL事务回滚机制理解

MySQL事务回滚机制是数据库管理系统中确保数据一致性和完整性的重要组成部分。当事务中的操作失败或需要撤销时,回滚机制能够确保所有已进行的数据修改不会生效,从而将数据库恢复到事务开始之前的状态。以下是关于MySQL事务回滚机制的详细理解:
MySQL事务回滚机制理解
2024-10-20

SpringBoot数据层测试事务回滚的实现流程

这篇文章主要介绍了SpringBoot数据层测试事务回滚的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
2022-11-13

jdbc回滚事务怎么实现

JDBC中回滚事务的实现可以通过以下步骤完成:首先,确保数据库连接已经开启了事务。可以通过设置连接的自动提交属性为false来实现,示例代码如下:connection.setAutoCommit(false);在事务执行过程中,如果发生异常
2023-10-26

MySQL数据库事务原理及应用

目录1 事务的使用1.1 事务概念1.2 事务的提交1.3 事务的常见操作2 事务隔离2.1 事务并发时出现的问题2.2 事务隔离级别1 事务的使用1.1 事务概念事务就是一组DML语句组成,这些语句在逻辑上存在相关性,这一组DML语句
2023-04-27

关于Spring中@Transactional事务回滚的注意事项

这篇文章主要介绍了关于Spring中@Transactional事务回滚的注意事项,回滚(Rollback)指的是程序或数据处理错误,将程序或数据恢复到上一次正确状态的行为。回滚包括程序回滚和数据回滚等类型,需要的朋友可以参考下
2023-05-19

mysql数据库事务及隔离级别

事务的四大特性: 原子性: 事务是最小的执行单位,不允许分割。事务的原子性确保动作要么全部完成,要么完全不起作用; 一致性: 执行事务前后,数据保持一致,多个事务对同一个数据读取的结果是相同的; 隔离性: 并发访问数据库时,一个用户的事务不被其他事务所干扰,各
mysql数据库事务及隔离级别
2021-09-18

java事务回滚redis怎么处理

Java事务回滚时,Redis缓存数据处理至关重要。有乐观和悲观两种回滚策略,选择取决于应用程序需求。乐观回滚避免数据库交互,提高性能,而悲观回滚确保数据一致性。实施回滚时,使用Redis事务性特性、设置过期时间、利用回调函数和异步处理。监听回滚事件、使用非阻塞客户端和单元测试也有助于提升处理可靠性。
java事务回滚redis怎么处理
2024-04-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录