Django_信號

Django中提供了'信號調度',用於在框架執行操做時解耦。html

通俗來說,就是一些動做發生時,信號容許特定的發送者去提醒一些接受者。python

 

Django內置信號mysql

Model signals
    pre_init                    # django的model執行其構造方法前,自動觸發
    post_init                   # django的model執行其構造方法後,自動觸發
    pre_save                    # django的model對象保存前,自動觸發
    post_save                   # django的model對象保存後,自動觸發
    pre_delete                  # django的model對象刪除前,自動觸發
    post_delete                 # django的model對象刪除後,自動觸發
    m2m_changed                 # django的model中使用m2m字段操做第三張表(add,remove,clear)先後,自動觸發
    class_prepared              # 程序啓動時,檢測已註冊的app中model類,對於每個類,自動觸發

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內置的信號,僅需註冊指定信號,當程序執行相應操做時,自動觸發註冊函數。sql

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


def callback(sender, **kwargs):
    print("xxoo_callback")
    print(sender,kwargs)

xxoo.connect(callback)
# xxoo指上述導入的內容
from django.core.signals import request_finished
from django.dispatch import receiver

@receiver(request_finished)
def my_callback(sender, **kwargs):
    print("Request finished!")

例子:數據庫

例子,建立數據庫記錄,觸發pre_save和post_save信號

建立一個Django項目,配置好路由映射

models.py中的代碼:

from django.db import models

class UserInfo(models.Model):
    name=models.CharField(max_length=32)
    pwd=models.CharField(max_length=64)
views.py中的代碼:

from django.shortcuts import render,HttpResponse
from app01 import  models

def index(request):
    models.UserInfo.objects.create(name="mysql",pwd="mysql123")
    return HttpResponse("ok")
項目的__init__.py文件中代碼:

from django.db.models.signals import pre_save,post_save

def pre_save_func(sender,**kwargs):

    print("pre_save_func")
    print("pre_save_msg:",sender,kwargs)

def post_save_func(sender,**kwargs):
    print("post_save_func")
    print("post_save_msg:",sender,kwargs)

pre_save.connect(pre_save_func)             # models對象保存前觸發callback函數
post_save.connect(post_save_func)           # models對象保存後觸發函數
建立一個index.html網頁,用瀏覽器打開這個項目,在服務端後臺打印信息以下:

pre_save_func
pre_save_msg: <class 'app01.models.UserInfo'> {'signal': <django.db.models.signals.ModelSignal object at 0x0000000002E62588>, 
'instance': <UserInfo: UserInfo object>, 'raw': False, 'using': 'default', 'update_fields': None}

post_save_func
post_save_msg: <class 'app01.models.UserInfo'> {'signal': <django.db.models.signals.ModelSignal object at 0x0000000002E62630>, 
'instance': <UserInfo: UserInfo object>, 'created': True, 'update_fields': None, 'raw': False, 'using': 'default'}
比較打印的結果,能夠看到models對象保存後,在打印信息裏包含一個"create=True"的鍵值對.

也能夠使用裝飾器來觸發信號,把上面__init__.py中的代碼修改:

from django.core.signals import request_finished
from django.dispatch import receiver

@receiver(request_finished)
def callback(sender, **kwargs):
    print("Request finished!")
則在本次請求結束後自動觸發callback函數,在後臺"request finished"這句話.
信號示例





 

自定義信號django

1.定義信號瀏覽器

新建一個項目,配置好路由,在項目根目錄下建立一個singal_test.py的文件,內容爲markdown

import django.dispatch

action=django.dispatch.Signal(providing_args=["aaaa","bbbb"])

 

2.註冊信號app

項目應用下面的__init__.py文件內容:框架

from singal_test import action

def pre_save_func(sender,**kwargs):

    print("pre_save_func")
    print("pre_save_msg:",sender,kwargs)
    
action.connect(pre_save_func)

 

3.觸發信號

views視圖函數內容:

from singal_test import action

action.send(sender="python",aaa="111",bbb="222")
用瀏覽器打開index.html網頁,後臺打印信息以下:

pre_save_func 
pre_save_msg: python {'signal': <django.dispatch.dispatcher.Signal object at 0x000000000391D710>, 'aaa': '111', 'bbb': '222'}
因爲內置信號的觸發者已經集成到Django中,因此會自動調用,而對於自定義信號須要在任意位置觸發

 
自定義信號示例2:
import django.dispatch
# 定義信號
pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"])

def callback(sender, **kwargs):
    print("callback")
    print(sender,kwargs)
#  註冊信號 
pizza_done.connect(callback)

# 觸發信號
pizza_done.send(sender='seven',toppings=123, size=456)
示例2
相關文章
相關標籤/搜索