Django ORM:两个日期之间的天数 timedelta 差异

Sat*_*nix 2 django django-orm

在此输入图像描述

我有一个数据库表,表示与给定产品相关的费用。

这些费用,假设它们是每天的,有一个from_date(开始日期)和to_date(结束日期)。to_date可以为空,因为这些费用可能仍在继续。

给定 2 个 Pythondatetimestart_dateend_date我需要在 ORM 中生成 期间花费的总费用my_product

>>> start_date
datetime.datetime(2021, 8, 20, 0, 0)

>>> end_date
datetime.datetime(2021, 9, 21, 0, 0)
Run Code Online (Sandbox Code Playgroud)

在这种情况下,预期输出应该是:

(-104 * (days between 08/20 and 08/25)) + (-113 * (days between 08/26 and 09/21)
Run Code Online (Sandbox Code Playgroud)

这是我到目前为止所得到的:

(
my_product.income_streams
    .values("product")
    .filter(type=IncomeStream.Types.DAILY_EXPENSE)
    .filter(add_to_commission_basis=True)
    .annotate(period_expenses=Case(
        When(Q(from_date__lte=start_date) & Q(to_date__lte=end_date),
 then=ExpressionWrapper( start_date - F('to_date'), output_field=IntegerField()))
    ), # Other When cases...
)
) # Sum all period_expenses results and you've got the solution
Run Code Online (Sandbox Code Playgroud)

这就是给我带来问题的原因:

then=ExpressionWrapper( start_date - F('to_date'), output_field=IntegerField())
Run Code Online (Sandbox Code Playgroud)

这个表达式总是返回 0(请注意,这就是为什么我什至不尝试乘以value:那是下一步)。

显然start_date - F('to_date')与“给我这两个日期之间的天数差”不同。

您可以在 Python 中使用timedelta. ORM 中的等价物是什么?

我尝试过ExtractDay

then=ExpressionWrapper( ExtractDay(start_date - F('to_date'))
Run Code Online (Sandbox Code Playgroud)

但我得到:django.db.utils.OperationalError: user-defined function raised exception

还尝试过DurationField

then=ExpressionWrapper(start_date - F('to_date'), output_field=DurationField())
Run Code Online (Sandbox Code Playgroud)

但这也返回零:datetime.timedelta(0)

Sat*_*nix 6

转换start_date为a解决了问题,下一步DateTimeField将差异转换为a 。DurationField

所以:

Cast(Cast(start_date, output_field=DateTimeField()) - F('to_date'), output_field=DurationField())
Run Code Online (Sandbox Code Playgroud)

这在任何数据库后端都可以正常工作,但是为了获得天数的差异,您需要将其包装在 中,如果您使用 SQLite,ExtractDay则会抛出异常。ValueError: Extract requires native DurationField database support.

如果您与 SQLite 绑定并且无法使用ExtractDay,您可以使用微秒,并通过除以 86400000000 手动将其转换为天

duration_in_microseconds=ExpressionWrapper(F('to_date') - (Cast(start_date, output_field=DateTimeField())), output_field=IntegerField())
Run Code Online (Sandbox Code Playgroud)

然后

.annotate(duration_in_days=ExpressionWrapper(F('period_duration_microseconds') / 86400000000, output_field=DecimalField())
Run Code Online (Sandbox Code Playgroud)

  • 嗨@Saturnix。我正在使用 mysql,当使用“ExtractDay”时,出现错误“Extract 需要本机 DurationField 数据库支持。”。任何想法? (2认同)