一 信號簡介
Django提供一種信號機制。其實就是觀察者模式,又叫發布-訂閱(Publish/Subscribe) 。當發生一些動作的時候,發出信號,然后監聽了這個信號的函數就會執行。
通俗來講,就是一些動作發生的時候,信號允許特定的發送者去提醒一些接受者。用於在框架執行操作時解耦。
二 Django內置信號
Model signals pre_init # django的modal執行其構造方法前,自動觸發 post_init # django的modal執行其構造方法后,自動觸發 pre_save # django的modal對象保存前,自動觸發 post_save # django的modal對象保存后,自動觸發 pre_delete # django的modal對象刪除前,自動觸發 post_delete # django的modal對象刪除后,自動觸發 m2m_changed # django的modal中使用m2m字段操作第三張表(add,remove,clear)前后,自動觸發 class_prepared # 程序啟動時,檢測已注冊的app中modal類,對於每一個類,自動觸發 Management signals pre_migrate # 執行migrate命令前,自動觸發 post_migrate # 執行migrate命令后,自動觸發 Request/response signals request_started # 請求到來前,自動觸發 request_finished # 請求結束后,自動觸發 got_request_exception # 請求異常后,自動觸發 Test signals setting_changed # 使用test測試修改配置文件時,自動觸發 template_rendered # 使用test測試渲染模板時,自動觸發 Database Wrappers connection_created # 創建數據庫連接時,自動觸發
Django 提供了一系列的內建信號,允許用戶的代碼獲得DJango的特定操作的通知。這包含一些有用的通知: django.db.models.signals.pre_save & django.db.models.signals.post_save 在模型 save()方法調用之前或之后發送。 django.db.models.signals.pre_delete & django.db.models.signals.post_delete 在模型delete()方法或查詢集的delete() 方法調用之前或之后發送。 django.db.models.signals.m2m_changed 模型上的 ManyToManyField 修改時發送。 django.core.signals.request_started & django.core.signals.request_finished Django建立或關閉HTTP 請求時發送。
三 內置信號的使用
對於Django內置的信號,僅需注冊指定信號,當程序執行相應操作時,自動觸發注冊函數:
方式1:
from django.core.signals import request_finished from django.core.signals import request_started from django.core.signals import got_request_exception from django.db.models.signals import class_prepared from django.db.models.signals import pre_init, post_init from django.db.models.signals import pre_save, post_save from django.db.models.signals import pre_delete, post_delete from django.db.models.signals import m2m_changed from django.db.models.signals import pre_migrate, post_migrate from django.test.signals import setting_changed from django.test.signals import template_rendered from django.db.backends.signals import connection_created
方式一:
#放到__init__里 from django.db.models.signals import pre_save import logging def callBack(sender, **kwargs): print(sender) print(kwargs) # 創建對象寫日志 logging.basicConfig(level=logging.DEBUG) # logging.error('%s創建了一個%s對象'%(sender._meta.db_table,kwargs.get('instance').title)) logging.debug('%s創建了一個%s對象'%(sender._meta.model_name,kwargs.get('instance').title)) pre_save.connect(callBack)
方式二:
from django.db.models.signals import pre_save from django.dispatch import receiver @receiver(pre_save) def my_callback(sender, **kwargs): print("對象創建成功") print(sender) print(kwargs)
四 自定義信號
a. 定義信號(一般創建一個py文件)(toppings,size 是接受的參數)
import django.dispatch pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"])
b. 注冊信號
def callback(sender, **kwargs): print("callback") print(sender,kwargs) pizza_done.connect(callback)
c. 觸發信號
from 路徑 import pizza_done pizza_done.send(sender='seven',toppings=123, size=456)
由於內置信號的觸發者已經集成到Django中,所以其會自動調用,而對於自定義信號則需要開發者在任意位置觸發。
練習:數據庫添加一條記錄時生成一個日志記錄。
