Базы данных - модели, разработка, реализация



         

Вложенные запросы - часть 2


Предикат NOT EXISTS обратно - истинен только тогда, когда подзапрос SubQuery пуст.

Обратите внимание, каким образом NOT EXISTS с вложенным запросом позволяет обойтись без операции разности отношений. Например, формулировка запроса со словом "все" может быть выполнена как бы с двойным отрицанием. Рассмотрим пример базы, которая моделирует поставку отдельных деталей отдельными поставщиками, она представлена одним отношением SP "Поставщики-детали" со схемой

  • SP (Номер_поставщика. номер_детали)
  • Р (номер_детали. наименование)
  • Вот каким образом формулируется ответ на запрос: "Найти поставщиков, которые поставляют все детали".

  • SELECT DISTINCT НОМЕР_ПОСТАВЩИКА
  • FROM SP SP1
  • WHERE NOT EXISTS
  • (SELECT номер_детали
  • FROM P
  • WHERE NOT EXISTS
  • (SELECT * FROM SP SP2
  • WHERE SР2.номер_поставщика=SР1.номер_поставщика AND
  • sр2.номер_детали = Р.номер_детали));
  • Фактически мы переформулировали этот запрос так: "Найти поставщиков таких, что не существует детали, которую бы они не поставляли". Следует отметить, что этот запрос может быть реализован и через агрегатные функции с подзапросом:

    88

  • SELECT DISTINCT Номер_поставщика
  • FROM SP
  • GROUP BY Номер_поставщика
  • HAVING Count(DISTINCT номер_детали) =
  • (SELECT Count( номер_детали)
  • FROM P)
  • В стандарте SQL92 операторы сравнения расширены до многократных сравнений с использованием ключевых слов ANY и ALL. Это расширение используется при сравнении значения определенного столбца со столбцом данных, возвращаемым вложенным запросом.

    Ключевое слово ANY, поставленное в любом предикате сравнения, означает, что предикат будет истинен, если хотя бы для одного значения из подзапроса предикат сравнения истинен. Ключевое слово ALL требует, чтобы предикат сравнения был бы истинен при сравнении со всеми строками подзапроса.

    Например, найдем студентов, которые сдали все экзамены на оценку не ниже чем "хорошо". Работаем с той же базой "Сессия", но добавим к ней еще одно отношение R4, которое характеризует сдачу лабораторных работ в течение семестра:

  • R1 = (ФИО, Дисциплина, Оценка);
  • R2 = (ФИО, Группа);
  • R3 = (Группы, Дисциплина )
  • R4 = (ФИО, Дисциплина, Номер_лаб_раб, Оценка);
  • Select R1.ФИО
  • From R1
  • Where 4 > = All (Select R1.Оценка
  • From R1 as R11
  • Where R1.Фио = R11.Фио)
  • Рассмотрим еще один пример:

    Выбрать студентов, у которых оценка по экзамену не меньше, чем хотя бы одна оценка по сданным им лабораторным работам по данной дисциплины:

  • Select R1.Фио
  • From R1
  • Where R1.Оценка >= ANY (Select Р4.Оценка
  • From R4
  • Where R1.Дисциплина = R4: Дисциплина AND R1.Фио = Р4.Фио)
  • 89

    87 :: 88 :: 89 :: Содержание




    Содержание  Назад  Вперед