Использование операторов набора
Операторы набора используются для объединения результатов двух (или более) операторов SELECT. Операторы SET, доступные в Oracle 11g, - это UNION, UNION ALL, INTERSECT и MINUS.
Оператор набора UNION возвращает объединенные результаты двух операторов SELECT. По сути, он удаляет дубликаты из результатов, т.е. для каждого дублированного результата будет указана только одна строка. Чтобы противостоять этому поведению, используйте оператор набора UNION ALL, который сохраняет дубликаты в окончательный результат. INTERSECT перечисляет только записи, общие для обоих запросов SELECT; оператор набора MINUS удаляет результаты второго запроса из вывода, если они также встречаются в результатах первого запроса. Операции над множествами INTERSECT и MINUS дают неповторимые результаты.
Все операторы SET имеют одинаковый приоритет среди них. Вместо этого во время выполнения запроса Oracle начинает вычисление слева направо или сверху вниз. Если используются явно круглые скобки, то порядок может отличаться, поскольку скобки будут иметь приоритет над висячие операторы.
Что нужно помнить -
Все участвующие операторы SELECT должны выбрать одинаковое количество столбцов. Имена столбцов, используемые в отображении, берутся из первого запроса.
Типы данных списка столбцов должны быть совместимы / неявно конвертируемы Oracle. Oracle не будет выполнять неявное преобразование типов, если соответствующие столбцы в запросах компонентов принадлежат к разным группам типов данных. Например, если столбец в запросе первого компонента имеет тип данных DATE, а соответствующий столбец во втором запросе компонента содержит данные типа CHAR, Oracle не будет выполнять неявное преобразование, но вызовет ошибку ORA-01790.
Для сортировки набора результатов необходимо использовать позиционный порядок. С помощью операторов Set нельзя упорядочивать индивидуальный набор результатов. ORDER BY может появиться один раз в конце запроса. Например,
Операторы UNION и INTERSECT коммутативны, т.е. порядок запросов не важен; это не меняет конечный результат.
Что касается производительности, UNION ALL показывает лучшую производительность по сравнению с UNION, поскольку ресурсы не тратятся зря на фильтрацию дубликатов и сортировку набора результатов.
Операторы множества могут быть частью подзапросов.
Операторы набора нельзя использовать в операторах SELECT, содержащих выражения коллекции TABLE.
LONG, BLOB, CLOB, BFILE, VARRAY или вложенная таблица не разрешены для использования в операторах Set. Предложение обновления не допускается с операторами set.
СОЮЗ
Когда несколько запросов SELECT объединяются с использованием оператора UNION, Oracle отображает объединенный результат всех составных запросов SELECT после удаления всех дубликатов и в отсортированном порядке (по умолчанию по возрастанию), не игнорируя значения NULL.
Рассмотрим следующие пять запросов, объединенных с помощью оператора UNION. Окончательный комбинированный набор результатов содержит значение из всех SQL-запросов. Обратите внимание на удаление дубликатов и сортировку данных.
SELECT 1 NUM FROM DUAL
UNION
SELECT 5 FROM DUAL
UNION
SELECT 3 FROM DUAL
UNION
SELECT 6 FROM DUAL
UNION
SELECT 3 FROM DUAL;
NUM
-------
1
3
5
6
Следует отметить, что столбцы, выбранные в запросах SELECT, должны иметь совместимый тип данных. Oracle выдает сообщение об ошибке при нарушении правила.
SELECT TO_DATE('12-OCT-03') FROM DUAL
UNION
SELECT '13-OCT-03' FROM DUAL;
SELECT TO_DATE('12-OCT-03') FROM DUAL
*
ERROR at line 1:
ORA-01790: expression must have same datatype as corresponding expression
СОЮЗ ВСЕ
UNION и UNION ALL похожи по своему функционированию с небольшой разницей. Но UNION ALL дает набор результатов без удаления дублирования и сортировки данных. Например, в приведенном выше запросе UNION заменяется на UNION ALL, чтобы увидеть эффект.
Рассмотрим запрос, показанный в разделе UNION. Обратите внимание на разницу в выводе, который создается без сортировки и дедупликации.
SELECT 1 NUM FROM DUAL
UNION ALL
SELECT 5 FROM DUAL
UNION ALL
SELECT 3 FROM DUAL
UNION ALL
SELECT 6 FROM DUAL
UNION ALL
SELECT 3 FROM DUAL;
NUM
-------
1
5
3
6
3
ПЕРЕСЕЧЕНИЕ
Используя оператор INTERSECT, Oracle отображает общие строки из обоих операторов SELECT, без дубликатов и данных, упорядоченных в отсортированном порядке (по умолчанию по возрастанию).
Например, приведенный ниже запрос SELECT извлекает зарплату, которая является общей для отделов 10 и 20. Согласно стандартам ISO SQL, INTERSECT превосходит другие по приоритету оценки операторов множества, но это все еще не включено в Oracle.
SELECT SALARY
FROM employees
WHERE DEPARTMENT_ID = 10
INTRESECT
SELECT SALARY
FROM employees
WHERE DEPARTMENT_ID = 20
SALARY
---------
1500
1200
2000
МИНУС
Оператор «Минус» отображает строки, которые присутствуют в первом запросе, но отсутствуют во втором запросе, без дубликатов и данных, расположенных в порядке возрастания по умолчанию.
SELECT JOB_ID
FROM employees
WHERE DEPARTMENT_ID = 10
MINUS
SELECT JOB_ID
FROM employees
WHERE DEPARTMENT_ID = 20;
JOB_ID
-------------
HR
FIN
ADMIN
Соответствие оператору SELECT
Могут быть сценарии, в которых составные операторы SELECT могут иметь различное количество и тип данных выбранных столбцов. Следовательно, для явного соответствия списку столбцов, столбцы NULL вставляются в недостающие позиции, чтобы соответствовать количеству и типу данных выбранных столбцов в каждом операторе SELECT. Для числовых столбцов можно также заменить ноль, чтобы соответствовать типу столбцов, выбранных в запросе.
В приведенном ниже запросе тип данных имени сотрудника (varchar2) и идентификатора местоположения (номер) не совпадают. Следовательно, выполнение следующего запроса вызовет ошибку из-за проблемы совместимости.
SELECT DEPARTMENT_ID "Dept", first_name "Employee"
FROM employees
UNION
SELECT DEPARTMENT_ID, LOCATION_ID
FROM departments;
ERROR at line 1:
ORA-01790: expression must have same datatype as corresponding expression
Явно сопоставить столбцы можно, подставив NULL вместо идентификатора местоположения и имени сотрудника.
SELECT DEPARTMENT_ID "Dept", first_name "Employee", NULL "Location"
FROM employees
UNION
SELECT DEPARTMENT_ID, NULL "Employee", LOCATION_ID
FROM departments;
Использование предложения ORDER BY в операциях SET
Предложение ORDER BY может появляться только один раз в конце запроса, содержащего составные операторы SELECT. Это означает, что отдельные операторы SELECT не могут содержать предложение ORDER BY. Кроме того, сортировка может быть основана на столбцах, которые появляются только в первом запросе SELECT. По этой причине рекомендуется сортировать составной запрос по позициям столбцов.
Приведенный ниже сводный запрос объединяет результаты из двух отделов и сортирует их по столбцу SALARY.
SELECT employee_id, first_name, salary
FROM employees
WHERE department_id=10
UNION
SELECT employee_id, first_name, salary
FROM employees
WHERE department_id=20
ORDER BY 3;