Это довольно распространенная проблема: создание отношения на лету без создания таблицы. Решения SQL для этой проблемы довольно неудобны. Один пример с использованием производной таблицы:
SELECT n.id
FROM
(SELECT 2 AS id
UNION SELECT 3
UNION SELECT 4
UNION SELECT 5
UNION SELECT 6
UNION SELECT 7) AS n
LEFT OUTER JOIN foos USING (id)
WHERE foos.id IS NULL;
Но это не очень хорошо масштабируется, потому что у вас может быть много значений вместо шести. Построить длинный список с одним UNION
, необходимым для значения, может стать утомительно.
Другое решение состоит в том, чтобы иметь под рукой универсальную таблицу из десяти цифр и многократно использовать ее для нескольких целей.
CREATE TABLE num (i int);
INSERT INTO num (i) VALUES (0), (1), (2), (3), (4), (5), (6), (7), (8), (9);
SELECT n.id
FROM
(SELECT n1.i + n10.i*10 AS id
FROM num AS n1 CROSS JOIN num AS n10
WHERE n1.i + n10.i*10 IN (2, 3, 4, 5, 6, 7)) AS n
LEFT OUTER JOIN foos USING (id)
WHERE foos.id IS NULL;
Я показываю внутренний запрос, генерирующий значения от 0 до 99, хотя это не обязательно для этого случая. Но в вашем списке могут быть значения больше 10. Дело в том, что с одной таблицей num
вы можете генерировать большие числа, не прибегая к очень длинным цепочкам с одной UNION
на значение. Также вы можете указать список желаемых значений в одном месте, что более удобно и удобно для чтения.