日期范围之间的PostgreSQL查询


126

我正在尝试查询我的postgresql数据库以返回日期在特定月份和年份的结果。换句话说,我想要一个月年的所有值。

到目前为止,我能够做到的唯一方法是这样的:

SELECT user_id 
FROM user_logs 
WHERE login_date BETWEEN '2014-02-01' AND '2014-02-28'

问题在于,我必须在查询表之前计算第一个日期和最后一个日期。有没有更简单的方法可以做到这一点?

谢谢


间隔是否总是一个月,或者是从15日开始到下个月的7月结束?
frlan 2014年

Answers:


202

使用日期(和时间),如果使用,许多事情会变得更简单>= start AND < end

例如:

SELECT
  user_id
FROM
  user_logs
WHERE
      login_date >= '2014-02-01'
  AND login_date <  '2014-03-01'

在这种情况下,您仍然需要计算所需月份的开始日期,但这应该以多种方式直接进行。

结束日期也得到了简化。只需添加一个月即可。不用担心28、30、31等。


该结构还具有能够维护索引使用的优点。


许多人可能会建议使用以下形式,但他们使用索引:

WHERE
      DATEPART('year',  login_date) = 2014
  AND DATEPART('month', login_date) = 2

这涉及计算表中每一行的条件(一次扫描),而不是使用索引来查找将匹配的行的范围(寻求范围)。


1
第一种方法需要花费数月(而不是数天)的外汇。2014-12-012015-01-01。第二个也可以被索引,但是我承认这EXTRACT()并不容易- 看起来更兼容。
pozs

1
您可以使用间隔来避免在应用程序中进行日期计算。例如:WHERE login_date> ='2014-02-01'AND login_date <'2014-02-01':: date + INTERVAL'1 month'在简化代码的同时,它仍使用索引。
巴斯威尔


22

万一有人落在这里...从8.1开始,您可以简单地使用:

SELECT user_id 
FROM user_logs 
WHERE login_date BETWEEN SYMMETRIC '2014-02-01' AND '2014-02-28'

从文档:

BETWEEN SYMMETRIC与BETWEEN相同,除了不要求AND左侧的参数小于或等于右侧的参数。如果不是,则会自动交换这两个参数,以便始终隐含一个非空范围。



By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.