95

根据我正在阅读的 Django 文档,似乎signals.py在 app 文件夹中是一个不错的起点,但我面临的问题是,当我创建信号pre_save并尝试从模型中导入类时,它与import在我的模型中。

# models.py

from django.contrib.auth.models import User
from django.db import models
from django.utils.translation import gettext as _
from signals import *

class Comm_Queue(CommunicatorAbstract):
    queue_statuses = (
        ('P', _('Pending')),
        ('S', _('Sent')),
        ('E', _('Error')),
        ('R', _('Rejected')),
    )
    status          = models.CharField(max_length=10, db_index=True, default='P')
    is_html         = models.BooleanField(default=False)
    language        = models.CharField(max_length=6, choices=settings.LANGUAGES)
    sender_email    = models.EmailField()
    recipient_email = models.EmailField()
    subject         = models.CharField(max_length=100)
    content         = models.TextField()

# signals.py

from django.conf import settings
from django.db.models.signals import pre_save
from django.dispatch import receiver
from models import Comm_Queue

@receiver(pre_save, sender=Comm_Queue)
def get_sender_email_from_settings(sender, **kwargs):
    obj=kwargs['instance']
    if not obj.sender_email:
        obj.sender_email='%s' % settings.ADMINS[0][1]

这段代码不会运行,因为我Comm_Queue在里面导入signals.py,我也在里面导入信号models.py

谁能建议我如何解决这个问题?

问候

4

8 回答 8

217

如果您使用的是 Django<=1.6,我建议您使用 Kamagatos 解决方案:只需在模型模块的末尾导入您的信号即可。

对于 Django 的未来版本(>=1.7),推荐的方法是在应用程序的 config ready()函数中导入信号模块:

my_app/apps.py

from django.apps import AppConfig

class MyAppConfig(AppConfig):
    name = 'my_app'

    def ready(self):
        import my_app.signals

my_app/__init__.py

default_app_config = 'my_app.apps.MyAppConfig'
于 2014-02-06T19:21:55.363 回答
68

原始答案,对于 Django < 1.7:

signals.py您可以通过在应用程序__init__.py文件中导入来注册信号:

# __init__.py
import signals

这将允许models.pysignals.py没有循环导入错误的情况下导入。

这种方法的一个问题是,如果您使用的是coverage.py,它会弄乱覆盖结果。

相关讨论

编辑:对于 Django >= 1.7:

自从 AppConfig 被引入以来,推荐的导入信号的方式就在它的init()函数中。有关更多详细信息,请参阅 Eric Marcos 的回答

于 2013-01-07T08:28:18.790 回答
27

要解决您的问题,您只需在模型定义后导入 signals.py。就这样。

于 2012-03-24T12:44:09.123 回答
6

我还将信号放在 signals.py 文件中,并且还有这个加载所有信号的代码片段:

# import this in url.py file !

import logging

from importlib import import_module

from django.conf import settings

logger = logging.getLogger(__name__)

signal_modules = {}

for app in settings.INSTALLED_APPS:
    signals_module = '%s.signals' % app
    try:
        logger.debug('loading "%s" ..' % signals_module)
        signal_modules[app] = import_module(signals_module)
    except ImportError as e:
        logger.warning(
            'failed to import "%s", reason: %s' % (signals_module, str(e)))

这是针对项目的,我不确定它是否适用于应用程序级别。

于 2012-03-08T09:28:30.777 回答
6

在旧的 Django 版本中,将信号放在__init__.py或者可能放在里面models.py(尽管最终模型对我的口味来说会很大)。

使用 Django 1.9,我认为最好将信号放在一个signals.py文件中并使用apps.py.

应用程序.py:

from django.apps import AppConfig


class PollsConfig(AppConfig):
    name = 'polls'

    def ready(self):
        from . import signals  # NOQA

您也可以在模型中的另一个文件夹中划分您的信号signals.py,但对我来说这只是过度工程。看看放置信号handlers.pysignals

于 2017-01-12T13:26:28.710 回答
4

这仅适用于您将信号放在单独的signals.py文件中

完全同意@EricMarcos 的回答,但应该指出,django文档明确建议不要使用 default_app_config 变量(尽管它没有错)。对于当前版本,正确的方法是:

my_app/apps.py

from django.apps import AppConfig

class MyAppConfig(AppConfig):
    name = 'my_app'

    def ready(self):
        import my_app.signals

设置.py

(确保您在已安装的应用程序中不仅有您的应用程序名称,而且还有 AppConfig 的相对路径)

INSTALLED_APPS = [
    'my_app.apps.MyAppConfig',
    # ...
]
于 2019-05-14T16:31:04.903 回答
3

我猜您这样做是为了注册您的信号,以便在某处找到它们。我只是把我的信号正常地放在一个models.py文件中。

于 2011-08-18T23:38:37.723 回答
1

另一种方法是从以下位置导入回调函数signals.py并将它们连接起来models.py

信号.py

def pre_save_callback_function(sender, instance, **kwargs):
    # Do stuff here

模型.py

# Your imports here
from django.db.models.signals import pre_save
from yourapp.signals import pre_save_callback_function

class YourModel:
    # Model stuff here
pre_save.connect(pre_save_callback_function, sender=YourModel)

ps:导入YourModelsignals.py产生递归;使用sender, 代替。

Ps2:在回调函数中再次保存实例会产生递归。您可以在.save方法中创建一个控制参数来控制它。

于 2012-03-21T18:57:59.160 回答