0

我有一个预先存在的“事件”表,其中包括一个 event_datetime 列和一个 event_duration_minutes 列。这两列分别是日期时间和整数。

我最近遇到了在事件结束时触发电子邮件的需要。为了实现这一点,我正在创建一个周期性工作,它每 10 分钟扫描一次数据库,以查找最近完成的事件。因此,我开始编写查询以查找在给定时间窗口内结束的所有事件,但由于每条记录的持续时间字段的可变性质,查询一直在逃避我。

我认为最好的方法是添加一个 end_time 列,然后自动将其设置为 start_time + 持续时间。这是正确的方法吗?

无论如何,我对如何进行查询完全空白 - 有没有一种简单的方法来处理这些情况?还是对此类查询的普遍需求表明您的数据库需要一些工作?

编辑 - 这是在 postgres 9.2 上。这是我的 rails schema.rb 中的模式:

create_table "events", :force => true do |t|
  t.string   "title"
  t.text     "details"
  t.datetime "created_at",             :null => false
  t.datetime "updated_at",             :null => false
  t.datetime "event_datetime"
  t.integer  "instructor_id"
  t.integer  "event_duration_minutes"
  t.datetime "started_at_time"
end
4

1 回答 1

2

通常的解决方案是在计算的结束时间上创建一个表达式索引。然后,您可以轻松地对其进行索引搜索。

最简单的方法是定义一个简单的 SQL 函数来执行计算,然后在表达式索引和要使用索引的查询中使用它。

例如:

CREATE OR REPLACE FUNCTION startduration_to_end(starttime timestamp, duration_minutes integer) RETURNS timestamp AS $$
SELECT $1 + $2 * INTERVAL '1' MINUTE';
$$ LANGUAGE sql;

然后:

CREATE INDEX mytable_end_index ON mytable 
( (startduration_to_end(event_datetime,event_duration_minutes ) );

根据查询模式,您可能希望在开始时间和结束时间设置另一个,但如果您的表具有较高的更改率,请记住维护索引的成本:

CREATE INDEX mytable_end_index ON mytable 
(event_datetime, (startduration_to_end(event_datetime,event_duration_minutes ) );

您可以使用您的函数使用这些索引进行查询startduration_to_end,例如:

SELECT *
FROM mytable t
WHERE startduration_to_end(t.event_datetime,t.event_duration_minutes) BETWEEN current_timestamp - INTERVAL '10' MINUTE AND current_timestamp;

由于执行时间的变化等,您确实应该扫描足够多的搜索重叠的索引,或者搜索比上次搜索的确切时间戳更新的所有内容,而不是最后 10 分钟。

于 2013-05-13T03:30:12.247 回答