python - 为什么我的查询集在 Django 中评估后会改变大小?

标签 python django django-queryset

我有一个简单的模型:

class Expense(models.Model):
    ...
    price = models.DecimalField(decimal_places=2, max_digits=6)
    is_fixed = models.BooleanField(default=False)

我正在尝试对 group_by 和聚合进行简单查询:

>>> from expenses.models import Expense
>>> from django.db.models import Sum
>>> qs = Expense.objects.values('is_fixed').annotate(total=Sum('price'))

我预计 qs 将有两条记录:

>>> qs.count()
2

但是当我评估它时,它返回 9!

>>> len(qs)
9

求值后,count 也开始返回 9:

>>> qs.count()
9

进一步说明:

>>> qs = Expense.objects.values('is_fixed').annotate(total=Sum('price'))
>>> qs.count() == len(qs)
False
>>> len(qs) == qs.count()
True
>>> qs = Expense.objects.values('is_fixed').annotate(total=Sum('price'))
>>> len(qs) == qs.count()
True

我不知道是什么导致了这种“异常”。你认为它可能是什么?

更新: 以下是 bdbd 要求的一些数据:

[
    {
        "price": 9.11,
        "category": "HOUSE",
        "created_at": "14/08/2021",
        "source": "BANK_SLIP",
        "is_fixed": True
    },
    {
        "price": 600.0,
        "category": "GIFT",
        "created_at": "09/08/2021",
        "source": "BANK_TRANSFER",
        "is_fixed": False
    },
    {
        "price": 10.5,
        "category": "FOOD",
        "created_at": "08/08/2021",
        "source": "SETTLE_UP",
        "is_fixed": False
    },
    {
        "price": 28.0,
        "category": "GIFT",
        "created_at": "08/08/2021",
        "source": "SETTLE_UP",
        "is_fixed": False
    },
    {
        "price": 20.44,
        "category": "SUPERMARKET",
        "created_at": "07/08/2021",
        "source": "SETTLE_UP",
        "is_fixed": False
    },
    {
        "price": 8.5,
        "category": "SUPERMARKET",
        "created_at": "06/08/2021",
        "source": "CREDIT_CARD",
        "is_fixed": False
    },
    {
        "price": 726.0,
        "category": "HOUSE",
        "created_at": "04/08/2021",
        "source": "BANK_SLIP",
        "is_fixed": True
    },
    {
        "price": 34.8,
        "category": "HOUSE",
        "created_at": "04/08/2021",
        "source": "BANK_SLIP",
        "is_fixed": True
    },
    {
        "price": 43.97,
        "category": "HOUSE",
        "created_at": "04/08/2021",
        "source": "MONEY",
        "is_fixed": True
    },
    {
        "price": 4.09,
        "category": "SUPERMARKET",
        "created_at": "04/08/2021",
        "source": "CREDIT_CARD",
        "is_fixed": False
    },
    {
        "price": 168.0,
        "category": "OTHER",
        "created_at": "04/08/2021",
        "source": "BANK_SLIP",
        "is_fixed": False
    },
    {
        "price": 23.4,
        "category": "FOOD",
        "created_at": "04/08/2021",
        "source": "SETTLE_UP",
        "is_fixed": False
    },
    {
        "price": 169.0,
        "category": "OTHER",
        "created_at": "03/08/2021",
        "source": "BANK_SLIP",
        "is_fixed": False
    },
    {
        "price": 9.81,
        "category": "SUPERMARKET",
        "created_at": "03/08/2021",
        "source": "CREDIT_CARD",
        "is_fixed": False
    },
    {
        "price": 100.0,
        "category": "RECREATION",
        "created_at": "03/08/2021",
        "source": "MONEY",
        "is_fixed": False
    },
    {
        "price": 10.0,
        "category": "SUPERMARKET",
        "created_at": "02/08/2021",
        "source": "CREDIT_CARD",
        "is_fixed": False
    }
]

最佳答案

问题是因为我在 Meta 上设置了 ordering:

class Meta:
    ordering = ("-created_at",)

在 order_by 子句上设置正确的字段解决了问题:

>>> qs = Expense.objects.order_by("is_fixed").values("is_fixed").annotate(total=Sum("price"))
>>> len(qs) == qs.count()
True
>>> qs = Expense.objects.order_by("is_fixed").values("is_fixed").annotate(total=Sum("price"))
>>> qs.count() == len(qs)
True

关于python - 为什么我的查询集在 Django 中评估后会改变大小?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68734419/

相关文章:

mysql - Django 与 MySQL 和 UTF-8

python - Django:我可以在不过滤所有模型实例的情况下检查模型实例是否与过滤器匹配吗

python - 为什么打印计算结果比仅仅计算要慢得多?

python - 打开 COM 端口以在客户端进行串行通信的解决方法(最好在 Python 中)

python - 是否可以访问在标准输入中传递给 python 的 python 脚本的源代码?

python - scipy.stats 随机抽取之间的区别....rvs 和 numpy.random

除非指定了应用程序和特定的测试方法,否则 Django 不会使用 manage.py test 运行测试

python - Base64 编码 HTTP_AUTHORIZATION header DRF

Django 使用列表过滤查询集

Django:提交表单时,自动填充字段