我试图找到一种方法来实现自定义QuerySet和自定义Manager,而不致干涸.这就是我目前的情况:

class MyInquiryManager(models.Manager):
    def for_user(self, user):
        return self.get_query_set().filter(
                    Q(assigned_to_user=user) |
                    Q(assigned_to_group__in=user.groups.all())
                )

class Inquiry(models.Model):   
    ts = models.DateTimeField(auto_now_add=True)
    status = models.ForeignKey(InquiryStatus)
    assigned_to_user = models.ForeignKey(User, blank=True, null=True)
    assigned_to_group = models.ForeignKey(Group, blank=True, null=True)
    objects = MyInquiryManager()

这很好,直到我做了这样的事情:

inquiries = Inquiry.objects.filter(status=some_status)
my_inquiry_count = inquiries.for_user(request.user).count()

这立刻打破了一切,因为QuerySetManager的方法不同.我曾try 创建一个定制的QuerySet类,并在MyInquiryManager中实现它,但我最终复制了所有的方法定义.

我还找到了this snippet,但我需要将额外的参数传递到for_user,这样它就崩溃了,因为它严重依赖于重新定义get_query_set.

有没有办法在不重新定义QuerySetManager子类中的所有方法的情况下做到这一点?

推荐答案

Django has changed!在使用2009年编写的这个答案中的代码之前,一定要查看其他答案和Django文档,看看是否有更合适的解决方案.


我实现这一点的方式是将实际的get_active_for_account添加为自定义QuerySet的方法.然后,要让它在管理器之外工作,您可以简单地捕获__getattr__并相应地将其返回

为了使此模式可重用,我将Manager位提取到单独的模型管理器:

cus至m_queryset/models.py

from django.db import models
from django.db.models.query import QuerySet

class Cus至mQuerySetManager(models.Manager):
    """A re-usable Manager 至 access a cus至m QuerySet"""
    def __getattr__(self, attr, *args):
        try:
            return getattr(self.__class__, attr, *args)
        except AttributeError:
            # don't delegate internal methods 至 the queryset
            if attr.startswith('__') and attr.endswith('__'):
                raise
            return getattr(self.get_query_set(), attr, *args)

    def get_query_set(self):
        return self.model.QuerySet(self.model, using=self._db)

一旦获得了这些,在您的模型上,您所需要做的就是将QuerySet定义为自定义内部类,并将管理器设置为您的自定义管理器:

your_app/models.py

from cus至m_queryset.models import Cus至mQuerySetManager
from django.db.models.query import QuerySet

class Inquiry(models.Model):
    objects = Cus至mQuerySetManager()

    class QuerySet(QuerySet):
        def active_for_account(self, account, *args, **kwargs):
            return self.filter(account=account, deleted=False, *args, **kwargs)

在这种模式下,以下任何一种都会起作用:

>>> Inquiry.objects.active_for_account(user)
>>> Inquiry.objects.all().active_for_account(user)
>>> Inquiry.objects.filter(first_name='John').active_for_account(user)

更新如果您将其与自定义用户(AbstractUser)一起使用,则需要更改
从…

class Cus至mQuerySetManager(models.Manager):

from django.contrib.auth.models import UserManager

class Cus至mQuerySetManager(UserManager):
    ***

Django相关问答推荐

如何在Django中将字段及其数据从一个模型添加到另一个模型?

Urls.py中路径**kwargs的Django翻译?

Django: 无法将我的 comments 关联到特定产品

我如何获得已循环的每个项目的名称?

Django 模型:如何查找自动生成的字段列表

Django中的多对多关系?

防止 django 将_id附加到外键字段

Django:在模板中呈现表单字段时添加 CSS 类

如何在 django tests.py 中创建管理员用户

Python 和 Django OperationalError (2006, 'MySQL server has gone away')

如何使用 SQL 的IN等字段上的数组过滤 django 查询集?

如何将我的上下文变量传递给 Django 中的 javascript 文件?

django 管理员操作而不 Select 对象

如何在 Django/Python 中减go 两个日期?

ProgrammingError: 安装 Psycopg2 后,关系django_session不存在错误

如何测试 Django QuerySets 是否相等?

如何在 django 中使用更少的 css?

Django中reverse()和reverse_lazy()的区别

Django 的 Select .如何设置默认选项?

如何更改 ModelForm 中所有 Django 日期字段的默认小部件?