Как сделать подзапрос в sql
Подзапрос SQL: полное руководство
Подзапрос SQL — это запрос внутри другого запроса. Он используется в предложении WHERE или HAVING оператора SQL. Подзапросы позволяют указать результаты одного запроса в качестве аргумента в другом запросе.
Когда вы пишете SQL- запрос, вы можете указать параметр на основе результата другого запроса. Например, скажем, у вас есть список клиентов. Вы можете получить список клиентов, которые разместили заказы на сумму, превышающую определённую сумму. Информация о заказе и клиенте хранится в другой таблице.
Здесь на помощь приходит операция подзапроса SQL. Подзапросы, также известные как вложенные запросы, представляют собой запросы SELECT в предложении WHERE другого запроса SQL. Затем основной запрос SQL использует результат подзапроса.
В этом руководстве мы обсудим, как использовать подзапросы в SQL, со ссылкой на несколько примеров.
Подзапрос SQL: руководство
Подзапрос SQL — это запрос внутри другого запроса. Они используются для выполнения запроса, который зависит от результатов другого запроса. Подзапросы позволяют делать это без необходимости писать два отдельных запроса и копировать, и вставлять результаты. Подзапросы появляются в предложениях WHERE или HAVING.
Вот синтаксис подзапроса в инструкции SQL SELECT:
SELECT name
FROM products
WHERE supplier_id
IN (SELECT id FROM suppliers WHERE local = True);
Этот запрос выбирает названия всех продуктов, поставляемых местной компанией. Мы используем подзапрос в качестве входных данных для оператора SQL IN. Оператор SQL IN возвращает все записи из таблицы «products», где в результатах нашего подзапроса находится идентификатор supplier_id.
Мы можем использовать подзапросы в операторах INSERT, UPDATE и DELETE.
Подзапрос должен быть заключён в круглые скобки. Это отличает внутреннюю часть подзапроса от другого подзапроса.
Убедитесь, что вы выбрали соответствующие данные в своём подзапросе. В нашем подзапросе выше мы выбрали идентификатор из таблицы поставщиков. Если бы мы выбрали другой столбец, наш запрос не работал бы. Это потому, что наш основной запрос зависит от идентификатора поставщика.
Пример подзапроса SQL: SELECT
Предположим, вы хотите получить список студентов, занесённых в списки почёта, вы можете использовать подзапрос. Это предполагает, что информация о списках части находится в другой таблице.
Давайте воспользуемся примером, чтобы проиллюстрировать, как работают подзапросы SQL. Следующий запрос вернёт список всех клиентов, которые сделали заказ на сумму более 200 долларов:
SELECT * FROM customers
WHERE id IN (SELECT DISTINCT customer_id FROM orders
WHERE cost > 200);
Наш подзапрос возвращает следующее:
name | address | loyalty_plan | id | |
Katy | katy.l@gmail.com | Mountain View, CA | None | 4 |
John | john.p@outlook.com | Boston, MA | None | 1 |
В первой строке мы выбираем каждый столбец из нашей таблицы клиентов. Затем мы указываем, что хотим получить только тех клиентов, чей идентификатор клиента находится в подзапросе. Наш подзапрос выбирает все уникальные идентификаторы клиентов из нашей таблицы заказов, где стоимость товара превышает 200 долларов. Записи, соответствующие этим условиям, включают этих клиентов в результаты.
Подзапрос SQL: INSERT
Мы хотим вставить все записи клиентов, совершивших покупку на сумму более 200 долларов, в таблицу с именем high_value_customers. Для этого мы можем использовать подзапрос SQL и команду SQL INSERT :
INSERT INTO high_value_customers
SELECT * FROM customers
WHERE id IN (SELECT DISTINCT customer_id FROM orders
WHERE cost > 200);
Если мы запросим нашу таблицу high_value_customers, мы сможем увидеть всех наших ценных клиентов в одной таблице:
name | address | loyalty_plan | id | |
Katy | katy.l@gmail.com | Mountain View, CA | Gold | 4 |
John | john.p@outlook.com | Boston, MA | None | 1 |
Подзапрос SQL: UPDATE
Мы могли бы использовать подзапрос для обновления данных в таблице. Например, предположим, что мы хотим обновить планы лояльности всех клиентов, совершивших дорогостоящую покупку, до уровня Bronze. Мы могли бы использовать следующий запрос для выполнения этой операции:
UPDATE customers
SET loyalty_plan = ‘High Value’
WHERE id IN (SELECT DISTINCT customer_id FROM orders
WHERE cost > 200);
Когда мы запрашиваем нашу таблицу клиентов, мы видим, что планы лояльности для двух наших ценных клиентов были обновлены:
name | address | loyalty_plan | id | |
Katy | katy.l@gmail.com | Mountain View, CA | High Value | 4 |
John | john.p@outlook.com | Boston, MA | High Value | 1 |
Подзапрос SQL: DELETE
Вы можете использовать подзапросы с оператором SQL DELETE для удаления отдельных или нескольких строк в таблице.
Напишем команду, которая удаляет заказы ценных клиентов, которые мы переместили в новую таблицу в базе данных. Для этого нам нужно будет зависеть от подзапроса:
DELETE FROM orders
WHERE customer_id IN (SELECT id FROM customers
WHERE loyalty_plan = ‘High Value’);
Мы видим, что все заказы от наших ценных клиентов были удалены. Наш вложенный подзапрос возвращает:
id | item_name | cost | customer_id |
5 | Oak Chair | 100 | 3 |
Вывод
Подзапрос SQL — это запрос внутри другого запроса. Подзапросы позволяют вам зависеть от результата одного запроса в другом. Подзапросы указываются в предложениях HAVING или WHERE оператора SQL.
Также подзапросы упрощают выполнение запроса, который зависит от результата другого запроса. В этом руководстве мы обсудили, как написать подзапрос на сервере SQL. Мы также говорили о том, как использовать подзапросы в SQL- операторах SELECT, INSERT, UPDATE и DELETE.
Как сделать подзапрос в sql
В выражении SELECT мы можем вводить подзапросы четырьмя способами:
Использовать в условии в выражении WHERE
Использовать в условии в выражении HAVING
Использовать в качестве таблицы для выборки в выражении FROM
Использовать в качестве спецификации столбца в выражении SELECT
Рассмотрим некоторые из этих случаев. Например, получим все товары, у которых цена выше средней:
Или выберем всех покупателей из таблицы Customers, у которых нет заказов в таблице Orders:
Хотя в данном случае подзапросы прекрасно справляются со своей задачей, стоит отметить, что это не самый эффективный способ для извлечения данных из других таблиц, так как в рамках T-SQL для сведения данных из разных таблиц можно использовать оператор JOIN, который рассматривается в следующей теме.
Получение набора значений
При использовании ключевого слова ALL условие в операции сравнения должно быть верно для всех значений, которые возвращаются подзапросом. Например, найдем все товары, цена которых меньше чем у любого товара фирмы Apple:
Если бы мы в данном случае опустили бы ключевое слово ALL, то мы бы столкнулись с ошибкой.
Допустим, если подзапрос возвращает значения vl1, val2 и val3, то условие фильтрации фактически было бы аналогично объединению этих значений через оператор AND:
В тоже время подобный запрос гораздо проще переписать другим образом:
При применении ключевых слов ANY и SOME условие в операции сравнения должно быть истинным для хотя бы одного из значений, возвращаемых подзапросом. По действию оба этих оператора аналогичны, поэтому можно применять любое из них. Например, в следующем случае получим товары, которые стоят меньше самого дорого товара компании Apple:
И также стоит отметить, что данный запрос можно сделать проще, переписав следующим образом:
Подзапрос как спецификация столбца
Результат подзапроса может представлять отдельный столбец в выборке. Например, выберем все заказы и добавим к ним информацию о названии товара:
Подзапросы в команде INSERT
В команде INSERT подзапросы могут применяться для определения значения, которое вставляется в один из столбцов:
Подзапросы в команде UPDATE
В команде UPDATE подзапросы могут применяться:
В качестве устанавливаемого значения после оператора SET
Как часть условия в выражении WHERE
Так, увеличим количество купленных товаров на 2 в тех заказах, где покупатель Тоm:
Или установим для заказа цену товара, полученную в результате подзапроса:
Подзапросы в команде DELETE
В команде DELETE подзапросы также применяются как часть условия. Так, удалим все заказы на Galaxy S8, которые сделал Bob:
Подзапросы в SQL (вложенные запросы SQL)
Что такое подзапросы в SQL?
Подзапросы SQL-примеры
В этом разделе мы рассмотрим, как использовать подзапросы. У нас есть следующие две таблицы: ‘ student ‘ и ‘ marks ‘ с общим полем ‘ StudentID ‘:
Теперь нужно составить запрос, определяющий всех студентов, которые получают лучшие отметки, чем студент со StudentID — « V002 ». Но мы не знаем отметок студента « V002 ».
Два приведенных запроса определяют студентов, которые получают лучше оценки, чем студент StudentID « V002 » ( Abhay ).
Можно объединить эти два запроса, вложив один запрос в другой. Подзапрос — это запрос внутри круглых скобок. Рассмотрим подзапроса в SQL пример :
Графическое представление подзапроса SQL :
Подзапросы в SQL (вложенные запросы SQL): общие правила
Ниже приведен синтаксис подзапроса :
Подзапросы SQL (вложенные запросы SQL): рекомендации по использованию
Ниже приведен ряд рекомендаций, которым нужно следовать при использовании SQL подзапросов:
Подзапросы SQL (вложенные запросы SQL) — основные типы
Подзапросы SQL с инструкцией INSERT
Пример таблицы: orders
Подзапросы SQL с инструкцией UPDATE
Подзапросы SQL с инструкцией DELETE
Если нужно удалить заказы из таблицы « neworder », для которых advance_amount меньше максимального значения advance_amount из таблицы « orders », можно использовать следующий код SQL :
Пример таблицы: neworder
Пожалуйста, оставляйте ваши мнения по текущей теме статьи. Мы очень благодарим вас за ваши комментарии, лайки, подписки, дизлайки, отклики!
Дайте знать, что вы думаете по этой теме статьи в комментариях. За комментарии, подписки, лайки, дизлайки, отклики огромное вам спасибо!
Вложенные запросы (SQL Server)
В примерах из этой статьи используется база данных AdventureWorks2016. Образцы баз данных AdventureWorks можно скачать здесь.
Основы вложенных запросов
Вложенный запрос по-другому называют внутренним запросом или внутренней операцией выбора, в то время как инструкцию, содержащую вложенный запрос, называют внешним запросом или внешней операцией выбора.
Многие инструкции языка Transact-SQL, включающие подзапросы, можно записать в виде соединений. Другие запросы могут быть осуществлены только с помощью подзапросов. В языке Transact-SQL обычно не бывает разницы в производительности между инструкцией, включающей вложенный запрос, и семантически эквивалентной версией без вложенного запроса. Дополнительные сведения о том, как SQL Server обрабатывает запросы, см. в разделе Обработка инструкций SQL. Однако в некоторых случаях, когда проверяется существование, соединения показывают лучшую производительность. В противном случае для устранения дубликатов вложенный запрос должен обрабатываться для получения каждого результата внешнего запроса. В таких случаях метод работы соединений дает лучшие результаты.
Вложенный во внешнюю инструкцию SELECT запрос, имеет следующие компоненты:
Запрос SELECT вложенного запроса всегда заключен в скобки. Он не может включать предложения COMPUTE или FOR BROWSE и может включать предложение ORDER BY только вместе с предложением TOP.
Если таблица появляется только во вложенном запросе, а не во внешнем запросе, в этом случае столбцы данной таблицы не могут быть включены в выходные данные (список выборки внешнего запроса).
Инструкции, включающие вложенные запросы, обычно имеют один из следующих форматов:
В некоторых инструкциях языка Transact-SQL вложенный запрос может рассматриваться как отдельный запрос. Обычно результаты вложенного запроса подставляются во внешний запрос (хотя SQL Server может обрабатывать инструкции Transact-SQL с вложенными запросами и по-другому).
Существуют три основных типа подзапросов, которые:
Правила вложенных запросов
На вложенный запрос распространяются следующие ограничения:
Уточнение имен столбцов во вложенных запросах
В следующем примере столбец BusinessEntityID в предложении WHERE внешнего запроса неявно уточняется именем таблицы, используемой в предложении FROM внешнего запроса (Sales.Store). Ссылка на столбец CustomerID в списке выборки вложенного запроса уточняется именем таблицы с помощью предложения FROM вложенного запроса, то есть Sales.Customer.
Общее правило состоит в том, что имена столбцов в инструкции неявно уточняются именем таблицы, указанной в предложении FROM того же уровня вложенности. Если столбец не существует в таблице, на которую ссылается предложение FROM вложенного запроса, он неявно уточняется именем таблицы, указанной в предложении FROM внешнего запроса.
Вот как выглядит этот запрос с явно указанными неявными соглашениями:
Никогда не будет ошибочным явно указать имя таблицы; также всегда можно перекрыть неявные соглашения об именах таблиц полностью уточненными именами столбцов
Если столбец, на который есть ссылка во вложенном запросе, не существует в таблице, указанной в предложении FROM вложенного запроса, но существует в таблице, на которую ссылается предложение FROM внешнего запроса, запрос будет выполнен без ошибок. SQL Server неявно уточнит имя столбца во вложенном запросе с помощью имени таблицы внешнего запроса.
Множественные уровни вложенности
Каждый вложенный запрос, в свою очередь, может содержать один или более вложенных запросов. В инструкцию можно вложить любое количество вложенных запросов.
Следующий запрос осуществляет поиск сотрудников, занимающих должность менеджера по продажам.
Самый глубоко вложенный запрос возвращает идентификаторы указанных сотрудников. Запрос уровнем выше оперирует с полученными идентификаторами и возвращает контактные идентификаторы сотрудников. Наконец, во внешнем запросе по полученным контактным идентификаторам извлекаются имена сотрудников.
Этот запрос также можно выразить с помощью соединения:
Связанные вложенные запросы
Результат для нескольких запросов может быть получен путем выполнения одного вложенного запроса и подстановки полученного результата или результатов в предложение WHERE внешнего запроса. В запросах, содержащих коррелированные вложенные запросы (также называемые повторяющимися вложенными запросами), вложенный запрос зависит по значению от внешнего запроса. Это означает, что выполнение вложенного запроса повторяется по одному разу для каждой строки, которая может быть выбрана внешним запросом. Такой запрос получает по одной записи для имени и фамилии каждого сотрудника, который в таблице SalesPerson имеет сумму премиальных, равную 5000, с соответствующими идентификаторами сотрудников в таблицах Employee и SalesPerson.
Результатом является 0.00 ( Syed Abbas не получал премиальных, потому что не является менеджером по продажам), поэтому выполнение внешнего запроса приводит к следующему результату:
Коррелированные вложенные запросы могут также включать в предложение FROM функции с табличным значением, указывая для них в качестве аргументов столбцы таблиц из внешнего запроса. В этом случае для каждой строки внешнего запроса выполняется функция с табличным значением, как и в случае с вложенным запросом.
Типы вложенных запросов
Вложенные запросы могут быть указаны во многих местах:
Вложенные запросы с псевдонимами таблицы
Многие инструкции, где вложенный и внешний запросы ссылаются на одну и ту же таблицу, могут быть переформулированы как самосоединения (соединения таблицы с самой собой). Например, можно найти адреса сотрудников из конкретного региона с помощью вложенного запроса:
Можно также использовать самосоединение:
Псевдонимы таблиц e1 и e2 необходимы, так как соединенная сама с собой таблица выступает в двух ролях. Псевдонимы можно также использовать во вложенных запросах, где и внешний, и внутренний запросы ссылаются на одну и ту же таблицу.
При использовании явных псевдонимов таблицы понятно, что ссылка на Person.Address во вложенном запросе означает не то же, что и ссылка во внешнем запросе.
Вложенные запросы с ключевым словом IN
Результат вложенного запроса, в котором присутствует ключевое слово IN (или NOT IN ) — это список из нуля или более значений. После того как вложенный запрос вернул результат, он используется внешним запросом.
Следующий запрос ищет названия всех колес, произведенных компанией Adventure Works Cycles.
Эта инструкция выполняется в два шага. Сначала внутренний запрос возвращает номер идентификатора подкатегории по соответствию названию «Wheel» (17). Затем это значение подставляется во внешний запрос, который находит все названия изделий, имеющих соответствующие идентификаторы подкатегорий в столбце Production.Product.
Единственная разница в использовании соединения и вложенного запроса для этой и аналогичных задач заключается в том, что объединение позволяет включить в результат столбцы, содержащиеся в нескольких таблицах. Например: если нужно включить в результат название подкатегории, следует пользоваться соединением:
Следующий запрос ищет названия всех поставщиков, имеющих высокий кредитный рейтинг, у которых компания Adventure Works Cycles заказала как минимум 20 позиций, и средний срок поставки у которых не превышает 16 дней.
Выполняется внутренний запрос и возвращаются номера идентификаторов поставщиков, которые соответствуют определениям вложенного запроса. Затем выполняется внешний запрос. Обратите внимание, что в предложение WHERE как внутреннего, так и внешнего запроса может быть включено несколько условий.
При использовании соединения тот же запрос будет выражен так:
Соединение всегда может быть выражено в виде вложенного запроса. Вложенный запрос часто, но не всегда может быть выражен в виде соединения. Это происходит потому, что соединения симметричны: можно соединить таблицы A и B в любом порядке и получить одинаковый результат. Для вложенных запросов это не всегда справедливо.
Вложенные запросы с ключевым словом NOT IN
Вложенные запросы с ключевым словом NOT IN также возвращают список из нуля или более значений.
В следующем запросе выполняется поиск названий продуктов, не являющихся готовыми велосипедами.
Эту инструкцию нельзя преобразовать в соединение. Аналогичное соединение по неравенству имеет другой смысл: оно находит названия продуктов, которые принадлежат какой-либо подкатегории, отличной от готового велосипеда.
В следующем примере удваивается значение столбца ListPrice таблицы Production.Product. Вложенный запрос в предложении WHERE ссылается на таблицу Purchasing.ProductVendor для ограничения количества обновляемых строк таблицы Product только теми, у которых идентификатор BusinessEntity равен 1540.
Для ясности, если на одну и ту же таблицу ссылаются в других вложенных запросах, используйте псевдонимы целевой таблицы:
Вложенные запросы с операторами сравнения
Чтобы использовать подзапрос, начинающийся с немодифицированного оператора сравнения, необходимо достаточно хорошо знать свои данные и природу проблемы, чтобы быть уверенным, что вложенный запрос возвратит точно одно значение.
Однако если сотрудник Linda Mitchell работал более чем с одной территорией продаж, вы получите сообщение об ошибке. Вместо оператора сравнения = может использоваться формулировка IN (также может использоваться =ANY ).
Вложенные запросы, начинающиеся с немодифицированных операторов сравнения, часто включают агрегатные функции, потому что они возвращают одиночное значение. Например, следующая инструкция находит названия всех продуктов, у которых цена по прейскуранту больше, чем средняя цена по прейскуранту.
Так как вложенные запросы, начинающиеся с неизмененных операторов сравнения, должны возвращать одиночное значение, они не могут включать предложения GROUP BY или HAVING (за исключением случаев, когда достоверно известно, что предложение GROUP BY или HAVING возвратит одиночное значение). Например, следующий запрос находит продукты, оцененные выше, чем самый дешевый продукт, который находится в столбце ProductSubcategoryID 14.
Рассмотрим, например оператор сравнения >: >ALL будет означать «больше любого значения». Другими словами, это сравнение с максимальным значением. Например, >ALL (1, 2, 3) означает «больше 3». >ANY означает «больше по крайней мере одного значения», т. е. «больше минимума». Поэтому >ANY (1, 2, 3) означает «больше 1». Чтобы строка результата вложенного запроса с >ALL удовлетворяла условию, заданному внешним запросом, значение в столбце, для которого вводится вложенный запрос, должно быть больше каждого значения из списка, возвращаемого вложенным запросом.
Аналогичным образом, чтобы строка результата вложенного запроса с >ANY удовлетворяла условию, заданному внешним запросом, значение в столбце, для которого вводится вложенный запрос, должно быть больше хотя бы одного значения из списка, возвращаемого вложенным запросом.
Если вложенный запрос не возвращает значений, весь запрос не возвратит никаких значений.
Вот результирующий набор, возвращаемый любым из этих запросов:
Однако оператор <>ANY отличается от NOT IN :
Например, следующий запрос отобразит заказчиков, находящихся на территории, где не работает ни один менеджер по продажам.
В результат включены все заказчики, кроме тех, чьим территориям продаж соответствует NULL, так как любая территория, назначенная заказчику, обслуживается менеджером по продажам. Внутренний запрос находит все территории продаж, обслуживаемые менеджерами по продажам, а затем для каждой территории внешний запрос находит заказчиков, которые ей не принадлежат.
По этой же причине, если использовать NOT IN в этом запросе, в результат не войдет ни один из заказчиков.
Вложенные запросы с ключевым словом EXISTS
Вложенный запрос, созданный с помощью ключевого слова EXISTS, имеет следующий синтаксис:
WHERE [NOT] EXISTS (subquery)
Следующий запрос ищет названия всех продуктов, которые находятся в подкатегории Wheels:
Обратите внимание на то, что вложенные запросы, введенные с помощью ключевого слова EXISTS, отличаются от других вложенных запросов следующим образом.
Вложенные запросы с ключевым словом NOT EXISTS
Например чтобы найти имена продуктов, не находящихся в подкатегории wheels:
Вложенные запросы, используемые вместо выражения
Следующий пример показывает, как можно использовать это улучшение. Запрос находит цены на все горные велосипеды, их среднюю цену и разницу между средней ценой и ценой каждого горного велосипеда.