微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

python-Django queryset优化:使用过滤器对_set进行反向查找

我有这些模型,我需要做一些计算并将其呈现给用户.我渲染了大约2-3k的行,这导致对数据库进行了4k个查询(如调试工具栏所示).有什么办法可以优化这个?我已经尝试过prefetch_related,但是它只是在已经完成的4k之上添加了另一个查询.

class Cart(models.Model):
    name = models.CharField(max_length=15)
    user = models.OnetoOneField(User)


    def sum_for_this(self, taxtype, tax):
        return self.carttax_set.filter(tax__type__name=taxtype,
                                       tax__tax=tax).aggregate(
            sum=Coalesce(Sum('tax_amount'), Value('0'))
        ).get('sum')


class TaxType(models.Model):
    name = models.CharField(max_length=10)


class Tax(models.Model):
    name = models.CharField(max_length=100)
    type = models.ForeignKey(TaxType)
    tax = models.DecimalField()


class CartTax(models.Model):
    cart = models.ForeignKey(Cart)
    tax = models.ForeignKey(Tax)
    base = models.IntegerField()
    tax_amount = models.IntegerField()

我在模板中所做的是:

{% for cart in cartlist %}
{{ cart.sum_for_this }}
{% endfor %}

我已经尝试过但没有效果

Cart.objects.prefetch_related('carttax_set').all()

方法def sum_for_this正在执行所有查询.

解决方法:

尝试对conditional expressions使用注释.查询将如下所示:

from django.db.models import Q, F, Sum, Case, When, IntegerField
from django.db.models.functions import Coalesce


cartlist = Cart.objects.annotate(
    sum=Coalesce(Sum(Case(
        When(Q(carttax__tax__type__name=taxtype) & Q(carttax__tax__tax=tax), then=F('carttax__tax_amount')),
        output_field=IntegerField()
        )), 0)
    )

并在模板中:

{% for cart in cartlist %}
{{ cart.sum }}
{% endfor %}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐