Django之信号

Posted zlixing

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Django之信号相关的知识,希望对你有一定的参考价值。

Django中提供了"信号调度",用于在框架执行操作时解耦.

一些动作发生的时候,系统会根据信号定义的函数执行相应的操作

Django中内置的signal

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,update),自动触发
class_prepared # 程序启动时,检测到已注册的model类,对于每一个类,自动触发

Managemeng_signals

pre_migrate                     # 执行migrate命令前,自动触发
post_migrate # 执行migrate命令后,自动触发

Request/response_signals

request_started                 # 请求到来前,自动触发
request_finished # 请求结束后,自动触发
got_request_exception # 请求异常时,自动触发

Test_signals

setting_changed                 # 配置文件改变时,自动触发
template_rendered # 模板执行渲染操作时,自动触发

Datebase_Wrapperd

connection_created              # 创建数据库连接时,自动触发

对于Django内置的信号,仅需注册指定信号,当程序执行相应操作时,系统会自动触发注册函数

例子,创建数据库记录,触发​​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对象保存后触发函数

 

也可以使用装饰器来触发信号,把上面​​__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"​​这句话.

 

自定义信号

1.定义信号

新建一个项目,配置好路由,在项目根目录下创建一个​​singal_test.py​​的文件,内容为

import django.dispatch

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

2.注册信号

项目应用下面的​​__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中,所以会自动调用,而对于自定义信号需要在任意位置触发

 

 

 

示例:

1.场景:新增标签的时候, 自动判断当前标签是否存在,存在则不需要处理, 不存在则加入标签库

models.py

from django.db.models.signals import post_save
from django.dispatch import receiver

class GameImg(models.Model):
app_id = models.IntegerField(null=False, default=0, verbose_name=u应用ID)
flag = models.CharField(max_length=16, default=, verbose_name=u"图片标识")
url = models.CharField(max_length=32, default=, verbose_name=u"图片地址")
remark = models.CharField(max_length=64, default=, verbose_name=u"备注")

def __str__(self):
return self.flag

class Meta:
db_table = game_img
verbose_name = verbose_name_plural = u"游戏图片表"


class GameImgFlag(models.Model):
flag = models.CharField(max_length=16, default=, verbose_name=u"图片标识")

def __str__(self):
return self.flag

class Meta:
db_table = game_img_flag
verbose_name = verbose_name_plural = u"游戏图片标识表"


@receiver(post_save, sender=GameImg)
def create_game_img(sender, instance, created, **kwargs):
is_exists = GameImgFlag.objects.filter(flag=instance.flag).first()
if not is_exists: # 新增flag
GameImgFlag.objects.create(flag=instance.flag)

 


以上是关于Django之信号的主要内容,如果未能解决你的问题,请参考以下文章

Django之信号

Django框架 之 信号

Django之信号

Django框架 之 信号

信号(Django信号Flask信号Scrapy信号)

flask,scrapy,django信号