如何使用 ExpressionWrapper 使用今天和 DateField 之间的差异来注释查询集

fru*_*ser 5 django annotate django-queryset datefield

我正在尝试根据 DateField 对查询集进行注释,如图所示。

我使用的是 Django 版本 1.8.12 和 MYSQL 版本 5.6.40。

尝试遵循如何用自创建以来的天数注释查询集,但这里它是一个 DateTimeField。下面的评论说“将其更改为 Value(now.date(), DateField()) - F('creation_date__date'),不起作用”

型号代码如下所示:

class Holding(models.Model):
    trans_date = models.DateField(_("Trans. Date"), null=False)
    ...
Run Code Online (Sandbox Code Playgroud)

给出错误持续时间的注释查询如下所示:

today = timezone.now().date()
testqs = Holding.objects.filter(id=1)
myqs = testqs.annotate(duration = ExpressionWrapper( Value(today, DateField()) - F('trans_date'), output_field=DurationField()))
Run Code Online (Sandbox Code Playgroud)

当我尝试打印时,我得到的只是“无”。打印差异以供参考。

for i in myqs:
    print i.duration, '-', today-i.trans_date

None - 1224 days, 0:00:00
None - 1206 days, 0:00:00
None - 1144 days, 0:00:00
None - 1051 days, 0:00:00
None - 1045 days, 0:00:00
Run Code Online (Sandbox Code Playgroud)

我希望持续时间是一个 timedelta 值,其中包含今天和 trans_date 之间的差异,而不是 None 。

sim*_*ick 4

这对我有用(带有 SQLite 的 Django 2.2.20):

import datetime as dt
from django import models
from django.db.models import F, ExpressionWrapper, fields
from django.db.models.functions import Now, TruncDate

class Test(models.Model):
    due_date=models.DateField(null=True, blank=True, default=dt.date.today)

due_in = ExpressionWrapper(
        F('due_date') - TruncDate(Now()),
        output_field=fields.DurationField())
qs = Test.objects.annotate(due_in=due_in)  # due_in will be dt.timedelta
Run Code Online (Sandbox Code Playgroud)

在一些具有本机 DurationField 支持的数据库中,我相信您可以执行以下操作以将天数提取为 int 字段,这将允许分组等,但这在 SQLite 中不受支持。

due_in_days = ExtractDay(due_in)
qs = Test.objects.annotate(due_in_days=due_in_days)  # result is int
Run Code Online (Sandbox Code Playgroud)