我有一个数据库表,其中包含代表年度重复事件的项目集。记录集按月和日存储。我经常需要检索与一系列日历日期相对应的事件记录。我正在使用 Django ORM 来处理记录,所以目前我将日期转换为相应的Q
对象(例如 Q(month=month, day=day)
和 OR 他们一起调用MyModel.objects.filter()
.
如果我的日期范围与新年相交,问题就来了。如果我想要从 2013 年 12 月 31 日到 2014 年 1 月 1 日的事件,我会执行以下操作:
MyModel.objects.filter(Q(month=12, day=31) | Q(month=1, day=1))
但我按以下顺序得到结果:
month = 1, day = 1
month = 12, day = 31
相反,我想按以下顺序获得结果:
month = 12, day = 31
month = 1, day = 1
出于不必要地使问题复杂化的原因,我不能简单地将查询分为两个查询,每年一个。我想进行一次查询并按所需顺序获取结果。如有必要,我可以重新制定查询。
我知道extra
应该有用,但我不太明白如何使用它。
更新:
为了更接近预期的解决方案,强加绝对顺序,我可以以某种方式将儒略日作为"计算"字段放入结果中,并按该字段排序。但是怎么做呢?
我有棘手的解决方案使用extra
和SQL CASE
语句:
start_month = 12
start_day = 31
end_month = 1
end_day = 1
query = (models.MyModel.objects.filter(Q(month=start_month, day=start_day) |
Q(month=end_month, day=end_day))
.extra(select={'order_me': '''CASE WHEN month*31+day < %s*31+%s
THEN (12+month)*31+day
ELSE (month)*31+day
END''' % (start_month, start_day)})
.extra(order_by=['order_me']))
一旦我添加了这个order_me
字段(这不好),我认为它应该在谓词中使用,而不是Q(...)|Q(...)
用于日期范围
query = (models.MyModel.objects.all()
.extra(select={'order_me': """CASE WHEN month*31+day < %s*31+%s
THEN (12+month)*31+day
ELSE (month)*31+day
END""" % (start_month, start_day) })
.extra(order_by=['order_me'])
.extra(where=['order_me < (12 + %s) * 31 + %s' % (end_month,
end_day)]))