Подзапросы для решения запросов
Подзапрос лучше всего определить как запрос внутри запроса. Подзапросы позволяют писать запросы, которые выбирают строки данных для критериев, которые фактически разрабатываются во время выполнения запроса во время выполнения. Более формально это использование оператора SELECT внутри одного из предложений другого оператора SELECT. Фактически, подзапрос может содержаться внутри другого подзапроса, который находится внутри другого подзапроса и так далее. Подзапрос также может быть вложен в инструкции INSERT, UPDATE и DELETE. Подзапросы должны быть заключены в круглые скобки.
Подзапрос можно использовать в любом месте, где разрешено выражение, при условии, что оно возвращает одно значение. Это означает, что подзапрос, который возвращает одно значение, также может быть указан как объект в списке предложения FROM. Это называется встроенным представлением, потому что, когда подзапрос используется как часть предложения FROM, он рассматривается как виртуальная таблица или представление. Подзапрос может быть помещен в предложение FROM, предложение WHERE или предложение HAVING основного запроса.
Oracle допускает максимальное вложение 255 уровней подзапроса в предложение WHERE. Не существует ограничений для вложенных подзапросов, выраженных в предложении FROM. На практике ограничение в 255 уровней на самом деле вовсе не является пределом, потому что редко можно встретить вложенные подзапросы, вложенные более трех или четырех уровней.
Оператор SELECT подзапроса очень похож на оператор SELECT, используемый для начала обычного или внешнего запроса. Полный синтаксис подзапроса:
( SELECT [DISTINCT] subquery_select_parameter
FROM {table_name | view_name}
{table_name | view_name} ...
[WHERE search_conditions]
[GROUP BY column_name [,column_name ] ...]
[HAVING search_conditions] )
Типы подзапросов
Single Row Sub Query: Подзапрос, который возвращает однострочный вывод. Они отмечают использование операторов сравнения одной строки при использовании в условиях WHERE.
Multiple row sub query: Подзапрос возвращает несколько строк вывода. Они используют операторы сравнения нескольких строк, такие как IN, ANY, ALL. Также могут быть подзапросы, возвращающие несколько столбцов.
Correlated Sub Query: Коррелированные подзапросы зависят от данных, предоставляемых внешним запросом. Этот тип подзапросов также включает подзапросы, в которых используется оператор EXISTS для проверки существования строк данных, удовлетворяющих указанным критериям.
Подзапрос одной строки
Однострочный подзапрос используется, когда результаты внешнего запроса основаны на одном неизвестном значении. Хотя этот тип запроса формально называется «однорядный», название подразумевает, что запрос возвращает несколько столбцов, но только одну строку результатов. Однако однострочный подзапрос может возвращать только одну строку результатов, состоящую только из одного столбца, для внешнего запроса.
В приведенном ниже запросе SELECT внутренний SQL возвращает только одну строку, то есть минимальную зарплату для компании. Он, в свою очередь, использует это значение для сравнения зарплат всех сотрудников и отображает только тех, чья зарплата равна минимальной зарплате.
SELECT first_name, salary, department_id
FROM employees
WHERE salary = (SELECT MIN (salary)
FROM employees);
Предложение HAVING используется, когда необходимо ограничить групповые результаты запроса на основании некоторого условия. Если результат подзапроса необходимо сравнить с групповой функцией, вы должны вложить внутренний запрос в предложение HAVING внешнего запроса.
SELECT department_id, MIN (salary)
FROM employees
GROUP BY department_id
HAVING MIN (salary) < (SELECT AVG (salary) FROM employees)
Подзапрос с несколькими строками
Многострочные подзапросы - это вложенные запросы, которые могут возвращать более одной строки результатов родительскому запросу. Многострочные подзапросы чаще всего используются в предложениях WHERE и HAVING. Поскольку он возвращает несколько строк, он должен обрабатываться операторами сравнения множеств (IN, ALL, ANY). Хотя оператор IN имеет то же значение, что обсуждалось в предыдущей главе, оператор ANY сравнивает указанное значение с каждым значением, возвращаемым подзапросом, в то время как ALL сравнивает значение с каждым значением, возвращаемым подзапросом.
Запрос ниже показывает ошибку, когда подзапрос одной строки возвращает несколько строк.
SELECT first_name, department_id
FROM employees
WHERE department_id = (SELECT department_id FROM employees WHERE LOCATION_ID = 100)
department_id = (select
*
ERROR at line 4:
ORA-01427: single-row subquery returns more than one row
Использование операторов нескольких строк
[> ВСЕ] Больше, чем максимальное значение, возвращенное подзапросом
[<ALL] Меньше наименьшего значения, возвращенного подзапросом
[<ANY] Меньше максимального значения, возвращенного подзапросом
[> ANY] Больше, чем наименьшее значение, возвращаемое подзапросом
[= ANY] Равно любому значению, возвращаемому подзапросом (то же, что и IN)
Вышеупомянутый SQL можно переписать с помощью оператора IN, как показано ниже.
SELECT first_name, department_id
FROM employees
WHERE department_id IN (SELECT department_id
FROM departments
WHERE LOCATION_ID = 100)
Обратите внимание, что в приведенном выше запросе IN сопоставляет идентификаторы отделов, возвращенные из подзапроса, сравнивает их с таковыми в основном запросе и возвращает имя сотрудника, удовлетворяющего условию.
Соединение было бы лучшим решением для вышеуказанного запроса, но в целях иллюстрации в нем использовался подзапрос.
Коррелированный подзапрос
В отличие от обычного подзапроса, где внешний запрос зависит от значений, предоставленных внутренним запросом, коррелированный подзапрос - это тот, в котором внутренний запрос зависит от значений, предоставленных внешним запросом. Это означает, что в коррелированном подзапросе внутренний запрос выполняется многократно, один раз для каждой строки, которая может быть выбрана внешним запросом.
Коррелированные подзапросы могут создавать таблицы результатов, которые отвечают на сложные вопросы управления.
Рассмотрим приведенный ниже запрос SELECT. В отличие от ранее рассмотренных подзапросов, подзапрос в этом операторе SELECT не может быть разрешен независимо от основного запроса. Обратите внимание, что внешний запрос указывает, что строки выбираются из таблицы сотрудников с псевдонимом e1. Внутренний запрос сравнивает столбец номера отдела сотрудника (DepartmentNumber) таблицы сотрудников с псевдонимом e2 с тем же столбцом для имени таблицы псевдонимов e1.
SELECT EMPLOYEE_ID, salary, department_id
FROM employees E
WHERE salary > (SELECT AVG(salary)
FROM EMP T
WHERE E.department_id = T.department_id)
Подзапрос с несколькими столбцами
Подзапрос с несколькими столбцами возвращает более одного столбца во внешний запрос и может быть указан в предложениях FROM, WHERE или HAVING внешнего запроса. Например, в приведенном ниже запросе показаны исторические данные о сотрудниках, чья текущая зарплата находится в диапазоне от 1000 до 2000 и которые работают в отделе 10 или 20.
SELECT first_name, job_id, salary
FROM emp_history
WHERE (salary, department_id) in (SELECT salary, department_id
FROM employees
WHERE salary BETWEEN 1000 and 2000
AND department_id BETWEEN 10 and 20)
ORDER BY first_name;
Когда подзапрос с несколькими столбцами используется в предложении FROM внешнего запроса, он создает временную таблицу, на которую могут ссылаться другие предложения внешнего запроса. Эта временная таблица более формально называется встроенным представлением. Результаты подзапроса обрабатываются так же, как и любая другая таблица в предложении FROM. Если временная таблица содержит сгруппированные данные, сгруппированные подмножества обрабатываются как отдельные строки данных в таблице. Рассмотрим предложение FROM в запросе ниже. Встроенное представление, сформированное подзапросом, является источником данных для основного запроса.
SELECT *
FROM (SELECT salary, department_id
FROM employees
WHERE salary BETWEEN 1000 and 2000);