Django資料庫--事務及事務回滾

来源:https://www.cnblogs.com/thomson-fred/archive/2018/12/30/10198528.html
-Advertisement-
Play Games

資料庫的讀寫操作中,事務在保證數據的安全性和一致性方面起著關鍵的作用,而回滾正是這裡面的核心操作。Django的ORM在事務方面也提供了不少的API。有事務出錯的整體回滾操作,也有基於保存點的部分回滾。本文將討論Django中的這兩種機制的運行原理。 Django利用django.db.transa ...


資料庫的讀寫操作中,事務在保證數據的安全性和一致性方面起著關鍵的作用,而回滾正是這裡面的核心操作。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='[email protected]')
    # 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='[email protected]')

    author_name = u'linghuchong'
    author = Author(name=author_name,age=26,email='[email protected]')
    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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 12 月閑暇的時間一直在更新 GearCase。通過不懈的努力,GearCase 今天迎來了一次中間版本的更新,這次的更新主要加入了 Springs 動畫組件,部分組件也添加了此組件的動畫效果。 > 除此之外還按照之前 jQuery 封裝實現過的輪播方式 的思路,重新製作了 Vue 輪子版本的組件, ...
  • 空對象模式 空對象模式是通過實現一個預設的無意義對象來避免null值出現, 簡單地說,就是為了避免在程式中出現null值判斷而誕生的一種常用設計方法. 舉個簡單的例子, 一個聽動物叫聲的模擬程式: 動物代碼: 聽叫聲的人: 現在我們想, 能不能把 animal 的空值判斷去掉呢?我們增加一個 Nul ...
  • 黑板模式的意圖 黑板模式是觀察者模式的一個擴展,知名度並不高,但使用的範圍卻非常廣. 其意圖為: 允許消息的讀寫同時進行, 廣泛的交互消息. 簡單的說, 黑板模式允許多個消息讀寫者同時存在, 消息的生產者和消費者完全分開. 這就像一個黑板, 任何一個老師(消息的生產者)都可以在上面書寫消息, 任何一 ...
  • 對象池模式 對象池模式, 或者稱為對象池服務, 其意圖為: 通過迴圈使用對象, 減少資源在初始化和釋放時的昂貴損耗(這裡的"昂貴"可能是時間效益(如性能), 也可能是空間效益(如並行處理), 在大多情況下, 指性能) 簡單的說, 在需要時,從池中提取,不用時,放回池中,等待下一個請求. 典型的例子是 ...
  • 雇工模式 雇工模式也叫僕人模式, 其意圖為: 雇工模式是行為模式的一種, 它為一組類提供通用的功能,而不需要類實現這些功能,它是命令模式的一種擴展. 其類圖如下: 其中角色如下: IServiced 代碼如下: Serviced1 和 Serviced2 實現了 IServiced, 代碼如下: S ...
  • 系統介紹: 1.系統採用主流的 SSM 框架 jsp JSTL bootstrap html5 (PC瀏覽器使用) 2.springmvc +spring4.3.7+ mybaits3.3 SSM 普通java web(非maven, 附贈pom.xml文件) 資料庫:mysql 3.開發工具:my ...
  • 行為類模式包括: 行為型模式涉及到演算法和對象間職責的分配 行為類模式關註如何管理對象的行為. 命令模式VS策略模式 策略模式和命令模式相似, 特別是命令模式退化時,比如無接收者(接收者非常簡單或者接收者是Java的基礎操作,無需專門寫一個接收者), 這種情況下, 命令模式和策略模式的類圖完全一樣. ...
  • 規格模式 規格模式 使用了策略模式,組合模式. 只不過將模式具體化了 規格模式將 與或非 操作進行了封裝, 實現了類似 SQL查詢語句的操作. 類圖如下: 其中的角色如下: 抽象規格書代碼: 組合規格書代碼: 與或非規格書代碼: 業務規格書: 場景類: 規格模式已經是一個非常具體的應用框架了, 遇到 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...