信号(signal)
作者:@skyflask
转载本文请注明出处:https://www.cnblogs.com/skyflask/p/9840404.html
目录
一、Django内置信号类型
1、Model signals
2、Management signals3、Request
/
response signals
4、Test signals
5、Database Wrappers
二、内置signal实例
1、pre_init(model初始化之前执行)
2、post_save(保存数据时触发执行)
三、自定义signal实例
a. 定义信号
b. 注册信号
c. 触发信号
d、执行结果
总结
一、Django内置信号类型
Django中提供了“信号机制”,用于在框架执行操作时解耦。通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者。
比如在数据库操作,插入一条数据之前和之后都写入日志。(审计工作)
这里装饰器就实现不了了,装饰器用在函数上,这里可能在一条代码前后,而且是每次都需要执行。如果你使用装饰器的话,你可能需要在Django源码里面修改save()函数,但是这样是不可取的。
Django早就帮你想到了,它在很多地方都放置了钩子。我们直接调用钩子就可以了。我们可以在信号里面注册很多个函数。触发信号时,会把信号里的函数执行一遍。
1、Model signals
1 2 3 4 5 6 7 8 | 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类,对于每一个类,自动触发 |
2、Management signals
1 2 | pre_migrate # 执行migrate命令前,自动触发 post_migrate # 执行migrate命令后,自动触发 |
3、Request
/
response signals
1 2 3 | request_started # 请求到来前,自动触发 request_finished # 请求结束后,自动触发 got_request_exception # 请求异常后,自动触发 |
4、Test signals
1 2 | setting_changed # 使用test测试修改配置文件时,自动触发 template_rendered # 使用test测试渲染模板时,自动触发 |
5、Database Wrappers
1 | connection_created # 创建数据库连接时,自动触发 |
二、内置signal实例
1、pre_init(model初始化之前执行)
由于signal必须在程序加载时就执行,所以我们需要将signal放在project的__init__.py文件下:
__init__.py文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | 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 ( "connection_created linked!" ) print (sender,kwargs) pre_init.connect(callback) |
启动app,signal函数立马执行:
2、post_save(保存数据时触发执行)
__init__.py文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | 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 ( "connection_created linked!" ) print (sender,kwargs) #使用Django内部定义的信号,在指定位置注入指定操作。比如这里我们在保存每一条记录之后做一个callback操作。 post_save.connect(callback) |
views.py文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | from django.core import serializers def index(request): if request.method = = 'GET' : print 'init前' ut = models.UserType.objects. filter ( id = 1 ).first() obj1 = models.User(username = 'singl11' , email = 'singl11@kingsoft.com' , user_type = ut) #测试先不执行save,看看效果 #obj1.save() obj = UserInfoForm() new_obj = models.User.objects. all () data = serializers.serialize( 'json' ,new_obj) print 'init 后' #print type(data) #print data return render(request, 'index.html' ,{ 'obj' :obj, 'new_obj' :new_obj}) elif request.method = = 'POST' : obj = UserInfoForm(request.POST) if obj.is_valid(): rt_dic = obj.cleaned_data username = rt_dic.get( 'username' ) email = rt_dic.get( 'email' ) user_type_id = rt_dic.get( 'user_type' ) ut = models.UserType.objects. filter ( id = user_type_id).first() models.User.objects.create(username = username,email = email,user_type = ut) return render(request, 'index.html' ,{ 'obj' :obj}) |
注意:这里先不执行save()。
结果如下:
从上可以看出,post_save()信号没有执行。
views.py中打开save()函数,执行结果如下:
1 2 | obj1 = models.User(username = 'singl11' , email = 'singl11@kingsoft.com' , user_type = ut) obj1.save() |
结果如下:
其他signal钩子一样,可以在对应的场景触发。
三、自定义signal实例
a. 定义信号
1 2 3 | #自定义signal import django.dispatch custom_signal = django.dispatch.Signal(providing_args = [ "args" , "kwargs" ]) |
b. 注册信号
1 2 3 4 5 | def callback1(sender, * * kwargs): print ( "callback custom_signal!" ) print (sender, kwargs) custom_signal.connect(callback1) |
c. 触发信号
1 2 3 4 5 6 7 8 9 | def index(request): if request.method = = 'GET' : #########触发signal############ from mform import custom_signal custom_signal.send(sender = 'xuequn' ,args1 = 123 , args2 = 456 ) obj = UserInfoForm() new_obj = models.User.objects. all () data = serializers.serialize( 'json' ,new_obj) return render(request, 'index.html' ,{ 'obj' :obj, 'new_obj' :new_obj}) |
d、执行结果
总结
由于内置信号的触发者已经集成到Django中,所以其会自动调用,而对于自定义信号则需要开发者在任意位置触发。
比如对系统状态阀值设置,到达某个状态,触发信号;数据库进行记录变更,我们可以进行某项操作,比如记录日志,审计功能。
这样只注册信号就可以,类似插拔式,降低程序耦合。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 上周热点回顾(3.3-3.9)
· AI 智能体引爆开源社区「GitHub 热点速览」