我有一些工作代码,但最近了解了相关管理器和反向查找,并想知道如何将它们应用于此代码:
我想利用相关管理器/反向查找的黑客方法是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]}
]