MySQL: предложение IN для дат между датой начала и окончания? - PullRequest
3 голосов
/ 04 ноября 2011

У меня есть список дат в формате 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 ]

2 голосов
/ 05 ноября 2011
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
0 голосов
/ 05 ноября 2011

Вы можете перевести свои спецификаторы месяца в начало и конец месяца (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' впарные DATE и DATETIME, а затем вычислите перекрытие.(Причина, по которой мы конвертируем eom в 'ГГГГ-ММ-ДД 23:59:59', заключается в том, что '2011-01' соответствует интервалу, starts_at '2011-01-01 01:00:00' (в противном случае eom будет принужден к 00:00:00 в первый день и не будет соответствовать нашему starts_at).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...