当前位置: 首页 > 面试题库 >

自定义QuerySet和Manager而不破坏DRY?

终育
2023-03-14
问题内容

我正在尝试找到一种方法来实现自定义QuerySet和自定义Manager而不破坏DRY。这是我到目前为止所拥有的:

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()

这会立即破坏所有内容,因为的QuerySet方法与的方法不同Manager。我尝试过创建一个自定义QuerySet类,并在中实现它MyInquiryManager,但最终我复制了所有方法定义。

我也发现此代码片段有效,但是我需要将额外的参数传递给,for_user因此它无法使用,因为它严重依赖于重新定义get_query_set

有没有一种方法可以在不重新定义子类QuerySetManager子类中的所有方法的情况下进行此操作?


问题答案:

我实现此目标的方法是将实际值添加get_active_for_account为自定义方法QuerySet。然后,要使其脱离管理器,你只需捕获__getattr__并相应地将其返回

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

custom_queryset / models.py

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

class CustomQuerySetManager(models.Manager):
    """A re-usable Manager to access a custom QuerySet"""
    def __getattr__(self, attr, *args):
        try:
            return getattr(self.__class__, attr, *args)
        except AttributeError:
            # don't delegate internal methods to 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)

一旦掌握了这些,就可以在模型上将QuerySeta定义为自定义内部类,并将管理器设置为自定义管理器:

your_app / models.py

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

class Inquiry(models.Model):
    objects = CustomQuerySetManager()

    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)

如果你将UPD与自定义用户(AbstractUser)一起使用,则需要

class CustomQuerySetManager(models.Manager):

from django.contrib.auth.models import UserManager

class CustomQuerySetManager(UserManager):
    ***


 类似资料:
  • 问题内容: 我有一个类似于以下示例(非常简化)的包装器类: 我可以这样使用它: 我以为可以通过更改为以下内容来优化并摆脱一个函数调用: 但是,我收到 TypeError:“包装”对象不支持索引 对于后者的版本线。 直接调用该方法(即)在两种情况下均有效… 为什么分配版本不允许索引? 问题答案: 必须在类上定义特殊方法(基本上是两端都带有两个下划线的任何方法)。特殊方法的内部查找过程完全跳过了实例字

  • 问题内容: 我想将列名修改为表中存在的新名称 但这里的问题我想手动修改或中的礼物。 有没有更好的方法可以做到这一点。 要一列正在使用此 我同样如何为做到这一点还是?没有打开每个脚本? 问题答案: 好吧,有很多第三方工具都承诺使用这种类型的“安全重命名”工具,其中一些是免费的,而有些则不是: MWillemse在回答中写道,ApexSQL为此提供了一个免费工具 RedGate有一个称为SQLProm

  • 问题内容: 我已经创建了文件夹并在其中初始化了virtualenv实例。 当我运行时,它将按原样显示已安装的软件包。 现在我想重命名为。 但是,当我跑步时 提示未安装pip。如何在不破坏环境的情况下重命名项目文件夹? 问题答案: 您需要调整安装以使用相对路径。为此提供了选项。从文档: 通常,环境与特定路径相关。这意味着您无法移动环境或将其复制到另一台计算机。您可以使用以下命令修复环境以使其可重定位

  • 问题内容: 我在postgres中使用表继承,但是我用来将数据分区到子表中的触发器的行为不太正确。例如,此查询返回nil,但我希望它返回新记录的。 如果将触发器函数的返回值从更改为,则会得到所需的行为,但是随后在数据库中插入了两个相同的行。这是有道理的,因为来自触发器函数的非空返回值会导致原始语句执行,而返回NULL会导致语句停止执行。唯一索引可能会中止第二次插入,但可能会引发错误。 有什么想法如

  • 问题内容: 我知道Go中没有析构函数,因为从技术上讲没有类。这样,我用来执行与构造函数相同的功能。但是,有没有办法在终止的情况下创建某些东西来模仿析构函数,例如使用关闭文件?现在,我只是打电话给我,但这有点荒唐,我认为设计很差。正确的方法是什么? 问题答案: 在Go生态系统中,存在一种处理包装了宝贵(和/或外部)资源的对象的惯用语:一种专门用于释放该资源的特殊方法,通常通过该机制进行 显式 调用。

  • .替换(...,Frag2,...) -->frag2 -->frag1