如何使用相关的管理器和反向查找来清理这个Django查询集



我有一些工作代码,但最近了解了相关管理器和反向查找,并想知道如何将它们应用于此代码:

我想利用相关管理器/反向查找的黑客方法是get_by_type_for_user(self, user):

class BadgeAssertionQuerySet(models.query.QuerySet):
    def get_user(self, user):
        return self.filter(user = user)
    def get_type(self, badge_type):
        return self.filter(badge__badge_type = badge_type)
    ...
class BadgeAssertionManager(models.Manager):
    def get_queryset(self):
    return BadgeAssertionQuerySet(self.model, using=self._db)
    ...
    def get_by_type_for_user(self, user):
        types = BadgeType.objects.all()
        qs = self.get_queryset().get_user(user)
        by_type =  [
                     {
                        'badge_type': t,
                        'list': qs.get_type(t)
                     } for t in types
                   ]
        return by_type

地点:

class BadgeAssertion(models.Model):
    badge = models.ForeignKey(Badge)
    user = models.ForeignKey(settings.AUTH_USER_MODEL)
    ...
    objects = BadgeAssertionManager()

:

class Badge(models.Model):
    name = models.CharField(max_length=50, unique=True)
    badge_type = models.ForeignKey(BadgeType)
    ....

:

class BadgeType(models.Model):
    name = models.CharField(max_length=50, unique=True)
    ...

我使用这个的目的是输出,对于一个特定的用户,他们已经获得的徽章,按徽章类型组织:

从观点:

context['badge_assertions_by_type'] = BadgeAssertion.objects.get_by_type_for_user(profile_user)
模板中的

片段(这是用户的配置文件页):

    {% for badges in badge_assertions_by_type %}
        {{badges.badge_type.name}}    
          {% for assertion in badges.list %}
              {{assertion.badge.name}}
              {{assertion.time_issued}}
          {% endfor %}
      {% endfor %}

你可以这样简化你的方法:

def get_badges_by_type_for_user(self, user):        
    qs = self.get_queryset()
             .select_related('badge', 'badge__badge_type')
             .filter(user=user)
    badges = defaultdict(list)
    for badge_assertion in qs:
        badges[badge_assertion.badge.badge_type].append(badge_assertion.badge)
    return badges

您不需要遍历所有可用的BadgeTypes,而只需将您获得的断言分组到徽章类型的桶中。因此,不需要对badge_type进行过滤。注意这里defaultdict()的使用。您可以尝试一下,看看它是否有效。我还没有亲自测试过。

结果结构在这里是不同的。它是:

{
    badge_type1: [badge1, badge2, ...],
    badge_type2: [badge3, badge4, ...]
}

代替:

[
    {'badge_type' : badge_type1, 'list': [badge1, badge2]},
    {'badge_type' : badge_type2, 'list': [badge3, badge4]}
]

相关内容

  • 没有找到相关文章

最新更新