1

我正在尝试使用单个查询来查找连接模型上的属性的最佳解决方案。

我有以下模型关系:

class Player(models.Model):
    first_name = models.CharField(max_length=30)
    last_name = models.CharField(max_length=30)
    season_team = models.ForeignKey(SeasonTeam, related_name='players', blank=True, null=True)
    leagues = models.ManyToManyField(League, related_name='players', through='PlayerLeague')


class League(models.Model):
    name = models.CharField()


class PlayerLeague(models.Model):
    player = models.ForeignKey(Player)
    league = models.ForeignKey(League)
    stamina = models.PositiveSmallIntegerField(_('Stamina'), default=100)


class Team(models.Model):
    name = models.CharField(max_length=30, validators=[
        RegexValidator(regex='^[a-zA-Z0-9\s]+$', message=_('name must contain only chars and numbers'), ), ])
    players = models.ManyToManyField(Player, blank=True, related_name='teams')
    league = models.ForeignKey(League, related_name='teams')

这是我希望在 get_context_data 中加载所有玩家以预加载耐力属性的视图

class FormationUpdateView(AjaxResponseMixin, FormationRequirementsMixin, UpdateView):
    model = Formation
    template_desktop = 'hockey/formation/form.html'
    template_mobile = 'hockey/formation/mobile/form.html'
    form_class = FormationForm

    def get_initial(self):
        self.initial = {'active': True}
        return self.initial.copy()

    def get_context_data(self, **kwargs):
        context = super(FormationUpdateView, self).get_context_data(**kwargs)
        team = Team.objects.select_related('league').get(formations=self.kwargs['pk'])
        context['players'] = Player.objects.filter(teams=team).all()
        context['league'] = team.league
        return context

    def get_queryset(self):
        return Formation.objects.select_related()

    def get_template_names(self):
        if 'Mobile' in self.request.META['HTTP_USER_AGENT']:
            return self.template_mobile
        return self.template_desktop

我的第一个解决方案是在 Player 类中添加一个 stamina 方法:

def stamina(self, league):
    p_l = self.leagues.through.objects.filter(league=league)[0]
    return p_l.stamina

我不喜欢这个解决方案,因为要查询每个玩家的耐力值。

感谢您的任何帮助

4

1 回答 1

2

您的模型有点令人费解,但是类似的东西应该可以最大程度地减少请求的数量

from django.db.models import Q, F, Prefetch

... 

def get_context_data(self, **kwargs)
     context = super(FormationUpdateView, self).get_context_data(**kwargs)
     team = (Team.objects
     .select_related('league')
     .prefetch_related(
         Prefetch(
             'players',
             queryset=(Player.objects.filter(leagues=F('leagues'))
                       .prefetch_related('playerleagues'))
         )
     ).get(formations=self.kwargs['pk']))
     context['players'] = team.players.all()
     context['league'] = team.league
     return context

在不再次访问数据库的情况下获得玩家的耐力。

players[0].playerleagues.all()[0].stamina
于 2016-08-21T21:47:45.337 回答