Django的信號機制詳解
Django提供一種信號機制。其實就是觀察者模式,又叫發布-訂閱(Publish/Subscribe) 。當發生一些動作的時候,發出信號,然后監聽了這個信號的函數就會執行。
Django內置了一些信號,比如:
django.db.models.signals.pre_save 在某個Model保存之前調用
django.db.models.signals.post_save 在某個Model保存之后調用
django.db.models.signals.pre_delete 在某個Model刪除之前調用
django.db.models.signals.post_delete 在某個Model刪除之后調用
django.core.signals.request_started 在建立Http請求時發送
django.core.signals.request_finished 在關閉Http請求時發送
我們要做的,就是注冊一個receiver函數。例如,如果要在每次請求完成之后,打印一行字。
可以使用回調的方式注冊:
# receiver def my_callback(sender, **kwargs):print("Request finished!")# connect from django.core.signalsimport request_finishedrequest_finished.connect(my_callback)
也可以使用裝飾器的方式注冊,下面這段代碼和上面完全是等價的。
from django.core.signalsimport request_finished from django.dispatchimport receiver@receiver(request_finished) def my_callback(sender, **kwargs):print("Request finished!")
receiver回調函數除了可以使用sender之外,還可以使用其他一些參數,比如針對pre_save函數:
sender:發送者(如果是pre_save的話,就是model class)
instance:實例
raw
using
update_fields
post_save()是一個比較實用函數,可以支持一些聯動的更新。而不必讓我們每次都寫在view里面。比如:有用戶提交了退款申請,我們需要把訂單的狀態修改成“已退款”的狀態。就可以使用信號機制,而不必在每處都修改。
@receiver(post_save, sender=RefundForm) deforder_state_update(sender, instance, created, **kwargs):instance.order.state = REFUNDINGinstance.order.save() # 這里,order是refundform的一個外鍵
當然,這里可以寫的更多更周全,例如退款單取消改回狀態等。
觀察者是非常實用的一個設計模式,Django也支持用戶 自定義 一些信號。
使用:
init.py
default_app_config = 'bug_management.apps.BugManagementConfig'
apps.py
from django.apps import AppConfigclass BugManagementConfig(AppConfig):name = 'bug_management'def ready(self):# signals are imported, so that they are defined and can be usedimport bug_management.signal
single.py
from django.shortcuts import render# Create your views here.from django.db.models import signals
from django.dispatch import receiverfrom bug_management.models import Bug_Detail
from .models import Bug_Detail#保存后 ,信號機制觸發 將jiraid 更改為正確id
@receiver(signals.post_save, sender=Bug_Detail)
def welcome_student(instance, created, **kwargs):print('保存了新bug')print('是否新創建了返回true和flase',created)print(instance)# 保存bug信息后 上傳jira 返回 jiraid 替換,instance.jiraid = '123'
?
?
?