django中實現事務的幾種方式
# 1 全局開啟事務---> 全局開啟事務,綁定的是http請求響應整個過程DATABASES = {'default': {#全局開啟事務,綁定的是http請求響應整個過程'ATOMIC_REQUESTS': True, }}from django.db import transaction# 局部禁用事務@transaction.non_atomic_requestsdef seckill(request):return HttpResponse('秒殺成功')# 2 一個視圖函數在一個事物中# fbv開啟from django.db import transaction@transaction.atomicdef seckill(request):return HttpResponse('秒殺成功')# cbv開啟from django.db import transactionfrom rest_framework.views import APIViewclass SeckillAPIView(APIView):@transaction.atomicdef post(self, request):pass# 3 局部使用事務
from django.db import transaction
def seckill(request):with transaction.atomic():pass # 都在一個事物中return HttpResponse('秒殺成功')
事物的回滾和保存點
# 1 普通事務操作(手動操作)
transaction.atomic() # 開啟事務
transaction.commit() # 提交事務
transaction.rollback() # 回滾事務# 2 可以使用上下文管理器來控制(自動操作)
with transaction.atomic(): # 自動提交和回滾# 3 保存點-開啟事務干了點事設置保存點1干了點事設置一個保存點2干了點事回滾到干完第二個事,回滾到保存點2'''
在事務操作中,我們還會經常顯式地設置保存點(savepoint)
一旦發生異常或錯誤,我們使用savepoint_rollback方法讓程序回滾到指定的保存點
如果沒有問題,就使用savepoint_commit方法提交事務
'''from .models import Book
from django.db import transaction
def seckill(request):with transaction.atomic():# 設置回滾點,一定要開啟事務sid = transaction.savepoint()print(sid)try:book = Book.objects.get(pk=1)book.name = '紅樓夢'book.save()except Exception as e:# 如發生異常,回滾到指定地方transaction.savepoint_rollback(sid)print('出異常了,回滾')# 如果沒有異常,顯式地提交一次事務transaction.savepoint_commit(sid)return HttpResponse('秒殺成功')
transaction.atomic() # 開啟事務
sid = transaction.savepoint() # 設置保存點
transaction.savepoint_rollback(sid) # 回滾到保存點
transaction.savepoint_commit(sid) #提交保存點
事務提交后,執行某個回調函數
# 有的時候我們希望當前事務提交后立即執行額外的任務,比如客戶下訂單后立即郵件通知賣家
###### 案例一##################
def send_email():print('發送郵件給賣家了')
def seckill(request):with transaction.atomic():# 設置回滾點,一定要開啟事務sid = transaction.savepoint()print(sid)try:book = Book.objects.get(pk=1)book.count = book.count-1book.save()except Exception as e:# 如發生異常,回滾到指定地方transaction.savepoint_rollback(sid)else:transaction.savepoint_commit(sid)#transaction.on_commit(send_email)transaction.on_commit(lambda: send_sms.delay('1898288322'))return HttpResponse('秒殺成功')##### 案例二:celery中使用###
transaction.on_commit(lambda: send_sms.delay('1898288322'))
django實現悲觀鎖樂觀鎖案例
# 線上賣圖書-圖書表 圖書名字,圖書價格,庫存字段-訂單表: 訂單id,訂單名字# 表準備class Book(models.Model):name = models.CharField(max_length=32)price = models.IntegerField() #count = models.SmallIntegerField(verbose_name='庫存')class Order(models.Model):order_id = models.CharField(max_length=64)order_name = models.CharField(max_length=32)# 使用mysql
DATABASES = {'default': {'ENGINE': 'django.db.backends.mysql','NAME': 'lqz','HOST': '127.0.0.1','PORT': '3306','USER': 'lqz','PASSWORD': '123',}
}# 創建lqz數據庫
原生mysql悲觀鎖
begin; # 開啟事務select * from goods where id = 1 for update; # 行鎖# order表中加數據update goods set stock = stock - 1 where id = 1; # 更新commit; #提交事務
orm實現上述
#1 使用悲觀鎖實現下單
@transaction.atomic # 整個過程在一個事物中---》改兩個表:book表減庫存,訂單表生成記錄
def seckill(request):# 鎖住查詢到的book對象,直到事務結束sid = transaction.savepoint() # 保存點# 悲觀鎖: select_for_update()# 加鎖了--》行鎖還是表鎖? 分情況,都有可能#book = Book.objects.select_for_update().filter(pk=1).first() # 加悲觀鎖,行鎖,鎖住當前行if book.count > 0:print('庫存可以,下單')# 訂單表插入一條Order.objects.create(order_id=str(datetime.datetime.now()), order_name='測試訂單')# 庫存-1,扣減的時候,判斷庫存是不是上面查出來的庫存,如果不是,就回滾time.sleep(random.randint(1, 4)) # 模擬延遲book.count=book.count-1book.save()transaction.savepoint_commit(sid) # 提交,釋放行鎖return HttpResponse('秒殺成功')else:transaction.savepoint_rollback(sid) #回滾,釋放行鎖return HttpResponse('庫存不足,秒殺失敗')
樂觀鎖秒殺--》庫存還有,有的人就沒成功
# 2 樂觀鎖秒殺--普通版
@transaction.atomic
def seckill(request):# 鎖住查詢到的book對象,直到事務結束sid = transaction.savepoint()book = Book.objects.filter(pk=1).first() # 沒加鎖count = book.countprint('現在的庫存為:%s' % count)if book.count > 0:print('庫存可以,下單')Order.objects.create(order_id=str(datetime.datetime.now()), order_name='測試訂單-樂觀鎖')# 庫存-1,扣減的時候,判斷庫存是不是上面查出來的庫存,如果不是,就回滾# time.sleep(random.randint(1, 4)) # 模擬延遲res = Book.objects.filter(pk=1, count=count).update(count=count - 1)if res >= 1: # 表示修改成功transaction.savepoint_commit(sid)return HttpResponse('秒殺成功')else: # 修改不成功,回滾transaction.savepoint_rollback(sid)return HttpResponse('被別人改了,回滾,秒殺失敗')else:transaction.savepoint_rollback(sid)return HttpResponse('庫存不足,秒殺失敗')