MySQL: предложение IN для дат между начальной и конечной датой?
У меня есть список дат в формате YYYY-MM
и у меня есть таблица, полная записей с starts_at
а также ends_at
колонка.
Я хочу найти все записи, для которых любая заданная дата в переданном списке находится в пределах starts_at
а также ends_at
спектр.
Итак, учитывая 2011-01, 2011-05, 2011-10
, Я хочу:
| id | starts_at | ends_at | title |
| 3 | 2010-12-05 00:00:00 | 2011-02-02 00:00:00 | something cool |
| 4 | 2011-03-14 00:00:00 | 2011-05-01 00:00:00 | something else really cool |
| 5 | 2011-10-31 00:00:00 | 2012-12-23 00:00:00 | argh! end of the world! not cool! |
... в то время как эти записи будут опущены:
| id | starts_at | ends_at | title |
| 6 | 2010-10-05 00:00:00 | 2010-12-02 00:00:00 | something uncool |
| 7 | 2011-03-14 00:00:00 | 2011-04-31 00:00:00 | something else really uncool |
| 8 | 2011-12-23 00:00:00 | 2013-01-01 00:00:00 | yay! we're still alive! cool! |
Как бы я написал это WHERE
условие в SQL?
Пояснение: я ищу решение на чистом SQL (я работаю с хранимой процедурой, поэтому, насколько я знаю, динамическое внедрение через какой-то другой язык, например PHP) не представляется возможным, и список дат передается на запрос как string
(Ввод формы HTML). Я хотел бы разбить его на последовательный BETWEEN
заявления, если бы я мог сделать это программно в SQL, но я понятия не имею, как это сделать.
По сути, мне нужен способ выразить следующую логику в чистом SQL:
$months = explode(',', '2011-01,2011-05,2011-10');
$q = "SELECT records.* FROM records WHERE";
foreach($months as $month) {
$q .= " '$month' BETWEEN records.starts_at AND records.ends_at OR";
}
$q = substr($q, 0, -3) . ';';
2 ответа
SELECT id, starts_at, ends_at, title
FROM yourtable
WHERE '2011-01-01' BETWEEN starts_at AND ends_at
OR '2011-05-01' BETWEEN starts_at AND ends_at
OR '2011-10-01' BETWEEN starts_at AND ends_at
Вы можете перевести спецификаторы месяца в начало и конец месяца (som
а также eom
ниже) и сделай сам
SET sql_mode='ansi';
SELECT neezer.*
FROM neezer
JOIN (SELECT m || '-01' AS som,
DATE_SUB(DATE_ADD(m || '-01', INTERVAL 1 MONTH), INTERVAL 1 SECOND) AS eom
FROM (SELECT '2011-01' AS m
UNION ALL
SELECT '2011-05'
UNION ALL
SELECT '2011-10') d) month_intervals
ON som <= ends_at AND eom >= starts_at;
Итак, мы конвертируем каждый '2011-01'
в пару ДАТА и DATETIME, а затем вычислите перекрытие. (Причина, по которой мы конвертируем eom
в "ГГГГ-ММ-ДД 23:59:59" так, что '2011-01'
соответствует интервалу, который starts_at
"2011-01-01 01:00:00" (в противном случае eom
будет принужден к 00:00:00 в первый день и не будет соответствовать нашему starts_at
).