• django的信号signals详解


    介绍:

    Django 提供一个了“信号分发器”机制,允许解耦的应用在框架的其它地方发生操作时会被通知到。 

    通俗而讲Django信号的工作原理,就是当某个事件发生的时候,会发出一个信号(signals), 而监听这个信号的函数(receivers)就会立即执行。

    就是A动了,给B发一个通知,然后B可以准备干点事情

    应用场景

    • 用户登陆后,系统向他发送最新动态信息。
    • 数据库数据发生变化后,实现缓存数据同步变化。
    • 订单中商品数量影响库存数量,即不同模型的联动更新。

    使用原则

    如果一个应用程序想要触发它引用的应用程序中的行为,则不要使用信号,而是直接导入它所需要的行为。

    如果一个应用程序想要触发依赖于该应用程序的应用程序中的行为,则可以在第二个应用程序中用接收器接收第一个应用程序发送给它的信号。

    如果信号接收器要处理大量I/O操作,也不要使用信号机制,因为它基于同步实现。

    用法

    Django常用内置信号函数介绍

    Model signals

        pre_init                    # django的modal执行其构造方法前,自动触发
        post_init                   # django的modal执行其构造方法后,自动触发
        pre_save                    # django的modal对象保存save()前,自动触发
        post_save                   # django的modal对象保存save()后,自动触发
        pre_delete                  # django的modal对象删除delete()前,自动触发
        post_delete                 # django的modal对象删除delete()后,自动触发
        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          # 创建数据库连接时,自动触发
    1. from django.core.signals import request_finished
    2. from django.core.signals import request_started
    3. from django.core.signals import got_request_exception
    4. from django.db.models.signals import class_prepared
    5. from django.db.models.signals import pre_init, post_init
    6. from django.db.models.signals import pre_save, post_save
    7. from django.db.models.signals import pre_delete, post_delete
    8. from django.db.models.signals import m2m_changed
    9. from django.db.models.signals import pre_migrate, post_migrate
    10. from django.test.signals import setting_changed
    11. from django.test.signals import template_rendered
    12. from django.db.backends.signals import connection_created

    基础用法

    需求,两个模型,当模型user创建实例的时候,打印出创建实例,当更新时,不提示

    1. from django.db import models
    2. from django.db.models import signals
    3. class User(models.Model):
    4. name = models.CharField(max_length=16)
    5. # 自定义函数,名字随便起
    6. def create_user(sender, instance, created, **kwargs):
    7. if created:
    8. print("New user created!")
    9. # 借用内置post_save进行二者表模型的连接,create_user被通知者,sender发送者
    10. signals.post_save.connect(create_user, sender=User)

    装饰器用法

    在实际应用中,一个更常用的方式是,使用@receiver装饰器,,实现发送者与监听函数的连接

    1. from django.db import models
    2. from django.db.models.signals import post_save
    3. from django.dispatch import receiver
    4. # User模型
    5. class User(models.Model):
    6. name = models.CharField(max_length=16)
    7. # post_save内置连接,sender发送者
    8. @receiver(post_save, sender=User)
    9. def create_user(sender, instance, created, **kwargs):
    10. # instance 更新
    11. # created 创建
    12. if created:
    13. print("New user created!")

    Django信号实现不同模型的联动更新

    我们有一个Profile模型,与User模型是一对一的关系。

    我们希望,创建User对象实例时,也创建Profile对象实例,而使用post_save更新User对象时,不创建新的Profile对象。

    这时我们就可以自定义,create_user_profile和save_user_profile两个监听函数,同时监听sender(User模型)发出的post_save信号。

    1. from django.db import models
    2. from django.db.models.signals import post_save
    3. from django.dispatch import receiver
    4. class User(models.Model):
    5. username = models.CharField(max_length=36)
    6. class Profile(models.Model):
    7. user = models.OneToOneField(User, on_delete=models.CASCADE)
    8. @receiver(post_save, sender=User)
    9. def create_user_profile(sender, instance, created, **kwargs):
    10. if created:
    11. Profile.objects.create(user=instance)
    12. @receiver(post_save, sender=User)
    13. def save_user_profile(sender, instance, **kwargs):
    14. instance.profile.save()

    signals拆分(推荐)

    介绍

    一个更好的方式,把自定义的信号监听函数,集中放在app对应文件夹下的signals.py文件里,便于后期集中维护。

    案例

    假如我们有个account的app,包含了User和Pofile模型,我们不仅需要在account文件夹下新建signals.py,还需要修改account文件下apps.py和__init__.py,以导入创建的信号监听函数。

    account/signals.py 

    1. from django.db.models.signals import post_save
    2. from django.dispatch import receiver
    3. from .models import User, Profile
    4. @receiver(post_save, sender=User)
    5. def create_user_profile(sender, instance, created, **kwargs):
    6. if created:
    7. Profile.objects.create(user=instance)
    8. @receiver(post_save, sender=User)
    9. def save_user_profile(sender, instance, **kwargs):
    10. instance.profile.save()

    account/apps.py

    1. from django.apps import AppConfig
    2. class AccountConfig(AppConfig):
    3. name = 'account'
    4. # 导入
    5. def ready(self):
    6. import account.signals

    account/__init__.py

    default_app_config = 'account.apps.AccountConfig'

    view

    1. # 测试
    2. def update_data(request):
    3. # 群更新会触发
    4. # query = ElasticNews.objects.all()
    5. # for item in query:
    6. # item.title = 'uuuuuuuuuuu'
    7. # item.save()
    8. # 单更新会触发
    9. # obj = ElasticNews.objects.filter(id=3).first()
    10. # obj.title = '99999999999'
    11. # obj.save()
    12. # create会触发
    13. # ElasticNews.objects.create(title='111', content='222',price='85')
    14. # update不会触发es与mysql同步,也不会触发信号post_save
    15. ElasticNews.objects.filter(id=1).update(title='圆梦去', content='222', price='85')
    16. return JsonResponse({'status': 200})

    本文参考 

    Django基础(31): 如何理解和正确使用Django信号(Signals) (qq.com)

    Django基础(31): 如何理解和正确使用Django信号(Signals) - 知乎 (zhihu.com)

    信号 | Django 文档 | Django (djangoproject.com)

    django:信号机制_dangfulin的博客-CSDN博客_django 信号

  • 相关阅读:
    JVM调优必备理论知识-GC Collector-三色标记
    Arthas--的使用
    高级架构之用户态网络协议栈TCP/IP设计
    RestClient查询文档
    keil4工程创建并进行流水灯实验
    23种设计模式(四)单例模式(阁瑞钛伦特软件-九耶实训)
    ESP8266 使用 DRV8833驱动板驱动N20电机
    锚框_的标定
    [ELK实战] 开发中的哪些坑
    Git入门实战教程之创建版本库
  • 原文地址:https://blog.csdn.net/qq_52385631/article/details/126531614