2

我有一个数据库表,其中包含代表年度重复事件的项目集。记录集按月和日存储。我经常需要检索与一系列日历日期相对应的事件记录。我正在使用 Django ORM 来处理记录,所以目前我将日期转换为相应的Q对象(例如Q(month=month, day=day),在对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应该很有用,但我不太明白如何使用它。

更新:

为了更接近预期的解决方案,强制执行绝对排序,我可以以某种方式将Julian Day作为“计算”字段放入结果中,并按该字段排序。但是怎么做呢?

4

1 回答 1

2

我有棘手的解决方案使用extra和 SQLCASE语句:

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)]))
于 2013-10-28T16:08:05.377 回答