|
|
|||||
|
> НАЗАД <
Глава 11S Q L СООТНЕСЕННЫЕ ПОДЗАПРОСЫ
В ЭТОЙ ГЛАВЕ, МЫ ПРЕДСТАВИМ ВАС ТИПУ
подзапроса о котором мы не говорили в Главе 10 - посвященной соотнесенному подзапросу. Вы узнаете как использовать соотнесенные подзапросы
в предложениях запросов WHERE и HAVING.
Сходства и различи между соотнесенными подзапросами и объединениями
будут обсуждаться далее, и вы сможете повысить ваше знание псевдонимов
и префиксов имени таблицы - когда они необходимы и как их использовать.
Когда вы используете подзапросы в SQL, вы можете обратиться к внутреннему запросу таблицы в предложении внешнего запроса FROM , сформировав - соотнесенный подзапрос. Когда вы делаете это, подзапрос выполняется неоднократно, по одному разу для каждой строки таблицы основного
запроса.
Соотнесенный подзапрос - один из большого количества тонких понятий
в SQL из-за сложности в его оценке.
Если вы сумеете овладеть им, вы найдете что он очень мощный, потому что
может выполнять сложные функции с помощью очень лаконичных указаний.
Например, имеется один способ найти всех заказчиков в порядках на 3-е
Октября ( вывод показывается в Рисунке 11.1 ):
В вышеупомянутом примере, "внутренний"(inner) и "внешний"(outer), это
псевдонимы, подробно обсужденным в =============== SQL Execution Log ============ | | | SELECT * | | FROM Customers outer | | WHERE 10/03/1990 IN | | (SELECT odate | | FROM Orders inner | | WHERE outer.cnum = inner.cnum); | | ============================================= | | cnum cname city rating snum | | ----- -------- ---- ------ ----- | | 2001 Hoffman London 100 1001 | | 2003 Liu San Jose 200 1002 | | 2008 Cisneros San Jose 300 1007 | | 2007 Pereira Rome 100 1004 | ============================================= Рисунок 11.1: Использование соотнесенного подзапроса ний запрос каждый раз будет выполнен, называется - текущей строкой-кандидатом. Следовательно, процедура оценки выполняемой соотнесенным подзапросом - это: 1. Выбрать строку из таблицы именованной в внешнем запросе. Это будет текущая строка-кандидат. 2. Сохранить значения из этой строки-кандидата в псевдониме с именем в предложении FROM внешнего запроса. 3. Выполнить подзапрос. Везде, где псевдоним данный для внешнего запроса найден ( в этом случае "внешний" ), использовать значение для текущей строки-кандидата. Использование значения из строки- кандидата внешнего запроса в подзапросе называется - внешней ссылкой. 4. Оценить предикат внешнего запроса на основе результатов подзапроса выполняемого в шаге 3. Он определяет - выбирается ли строка-кандидат для вывода. 5. Повторить процедуру для следующей строки-кандидата таблицы, и так далее пока все строки таблицы не будут проверены. В вышеупомянутом примере, SQL осуществляет следующую процедуру: 1. Он выбирает строку Hoffman из таблицы Заказчиков. 2. Сохраняет эту строку как текущую строку-кандидат под псевдонимом - "внешним". 3. Затем он выполняет подзапрос. Подзапрос просматривает всю таблицу Порядков чтобы найти строки где значение cnum поле - такое же как значение outer.cnum, которое в настоящее время равно 2001, - поле cnum строки Hoffmanа. Затем он извлекает поле odate из каждой строки таблицы Порядков для которой это верно, и формирует набор значений пол odate. 4. Получив набор всех значений пол odate, для пол cnum = 2001, он проверяет предикат основного запроса чтобы видеть имеется ли значение на 3 Октября в этом наборе. Если это так(а это так), то он выбирает строку Hoffmanа для вывода ее из основного запроса. 5. Он повторяет всю процедуру, используя строку Giovanni как строку-канди- дата, и затем сохраняет повторно пока каждая строка таблицы Заказчиков не будет проверена. Как вы можете видеть, вычисления которые SQL выполняет с помощью этих простых инструкций - это полный комплекс. Конечно, вы могли бы решить ту же самую проблему используя объединение, следующего вида ( вывод для этого запроса показывается в Рисунке 11.2 ): SELECT * FROM Customers first, Orders second WHERE first.cnum = second.cnum AND second.odate = 10/03/1990; Обратите внимание что Cisneros был выбран дважды, по одному разу для каждого порядка который он имел для данной даты. Мы могли бы устранить это используя SELECT DISTINCT вместо просто SELECT. Но это необязательно в варианте подзапроса. Оператор IN, используемый в варианте подзапроса, не делает никакого различи между значениями которые выбираются подзапросом один раз и значениями которые выбираются неоднократно. Следовательно DISTINCT необязателен. =============== SQL Execution Log ============ | | | SELECT * | | FROM Customers first, Orders second | | WHERE first.cnum = second.cnum | | (SELECT COUNT (*) | | FROM Customers | | WHERE snum = main.snum; | | ============================================= | | cnum cname | | ----- -------- | | 1001 Peel | | 1002 Serres | ============================================= Рисунок 11. 2 Использование объединения вместо соотнесенного подзапроса Предположим что мы хотим видеть имена и номера всех продавцов которые имеют более одного заказчика. Следующий запрос выполнит это для вас ( вывод показывается в Рисунке 11.3 ): SELECT snum, sname FROM Salespeople main WHERE 1 < ( SELECT COUNT (*) FROM Customers WHERE snum = main.snum ); Обратите внимание что предложение FROM подзапроса в этом примере не использует псевдоним. При отсутствии имени таблицы или префикса псевдонима, SQL может для начала принять, что любое поле выводится из таблицы с именем указанным в предложении FROM текущего запроса. Если поле с этим именем отсутствует( в нашем случае - snum ) в той таблице, SQL будет проверять внешние запросы. Именно поэтому, префикс имени таблицы обычно необходим в соотнесенных подзапросах - для отмены этого предположения. Псевдонимы также часто запрашиваются чтобы давать вам возможность ссылаться к той же самой таблице во внутреннем и внешнем запросе без какой-либо неоднозначности. =============== SQL Execution Log ============ | | | SELECT snum sname | | FROM Salespeople main | | WHERE 1 < | | AND second.odate = 10/03/1990; | | ============================================= | | cnum cname city rating snum | | ----- -------- ---- ------ ----- | | 2001 Hoffman London 100 1001 | | 2003 Liu San Jose 200 1002 | | 2008 Cisneros San Jose 300 1007 | | 2007 Pereira Rome 100 1004 | ============================================= Рисунок 11.3: Нахождение продавцов с многочисленными заказчиками
Иногда полезно выполнять запросы которые разработаны специально так
чтобы находить ошибки. Это всегда возможно при дефектной информации
которую можно ввести в вашу базу данных, и, если она введена, бывает
трудно ее определить. Следующий запрос не должен производить никако-
го вывода. Он просматривает таблицу Порядков чтобы видеть совпадают
ли пол snum и cnum в каждой строке таблицы Заказчиков и выводит каж-
дую строку где этого совпадения нет. Другими словами, запрос выясняет,
тот ли продавец кредитовал каждую продажу ( он воспринимает поле cnum,
как первичный ключ таблицы Заказчиков, который не будет иметь никаких
двойных значений в этой таблице ).
При использовании механизма справочной целостности ( обсужденного
в
Вы можете также использовать соотнесенный подзапрос основанный на
той же самой таблице что и основной запрос. Это даст вам возможность
извлечь определенные сложные формы произведенной информации.
Например, мы можем найти все порядки со значениями сумм приобретений выше среднего для их заказчиков ( вывод показан в Рисунке 11.4 ):
Рисунок 11.4: Соотнесение таблицы с собой
Конечно, в нашей маленькой типовой таблице, где большинство заказчиков
имеют только один порядок, большинство значений являются одновременно
средними и следовательно не выбираются. Давайте введем команду другим
способом ( вывод показывается в Рисунке 11.5 ):
Рисунок 11.5: Выбираются порядки которые > = средней сумме приобретений
для их заказчиков.
Различие, конечно, в том, что реляционный оператор основного предиката
включает значения которые равняются среднему ( что обычно означает
что они - единственные порядки для данных заказчиков ).
Также как предложение HAVING может брать подзапросы, он может брать и
соотнесенные подзапросы. Когда вы используете соотнесенный подзапрос
в предложении HAVING, вы должны ограничивать внешние ссылки к позициям которые могли бы непосредственно использоваться в самом предложе-
нии HAVING. Вы можете вспомнить из SELECT odate, SUM (amt) FROM Orders a GROUP BY odate HAVING SUM (amt) > ( SELECT 2000.00 + MAX (amt) FROM Orders b WHERE a.odate = b.odate );
Подзапрос вычисляет значение MAX для всех строк с той же самой датой
что и у текущей агрегатной группы основного запроса. Это должно быть
выполнено, как и ранее, с использованием предложения WHERE. Сам
подзапрос не должен использовать предложения GROUP BY или HAVING.
Как вы и могли предположить, соотнесенные подзапросы по природе близки
к объединениям - они оба включают проверку каждой строки одной таблицы
с каждой строкой другой ( или псевдонимом из той же ) таблицы.
Вы найдете что большинство операций которые могут выполняться с одним
из них будут также работать и с другим.
Однако имеется различие в прикладной программе между ними, такое как
вышеупомянутая потребность в использовании DISTINCT с объединением
и его необязательность с подзапросом. Имеются также некоторые вещи
которые каждый может делать так, как этого не может другой. Подзапросы,
например, могут использовать агрегатную функцию в предикате, дела
возможным выполнение операций типа нашего предыдущего примера в
котором мы извлекли порядки усредненные для их заказчиков.
Объединения, с другой стороны, могут выводить строки из обеих сравниваемых таблиц, в то врем как вывод подзапросов используется только в
предикатах внешних запросов. Как правило, форма запроса которая кажется наиболее интуитивной будет вероятно лучшей в использовании, но
при этом хорошо бы знать обе техники для тех ситуаций когда та или иная
могут не работать.
Вы можете поздравлять себя с овладением большого куска из рассмотренных
понятий в SQL - соотнесенного подзапроса. Вы видели как соотнесенный
подзапрос связан с объединением, а также, как его можно использовать с
агрегатными функциями и в предложении HAVING. В общем, вы теперь
узнали все типы подзапросов полностью.
Следующий шаг - описание некоторых SQL специальных операторов. Они
берут подзапросы как аргументы, как это делает IN, но в отличие от IN, они
могут использоваться только в подзапросах. Первый из их, представленный
в
1. Напишите команду SELECT использующую соотнесенный подзапрос,
которая выберет имена и номера всех заказчиков с максимальными
для их городов оценками.
2. Напишите два запроса которые выберут всех продавцов ( по их имени
и номеру ) которые в своих городах имеют заказчиков которых они не
обслуживают. Один запрос - с использованием объединения и один -
с соотнесенным подзапросом.
Которое из решений будет более изящным?
( Подсказка: один из способом это сделать, состоит в том, чтобы находить
всех заказчиков не обслуживаемых данным продавцом и определить,
находится ли каждый из них в городе продавца. )
( См.
> НАЗАД <
|
|||||