Перейти к оглавлению
               10. ВСТАВКА ОДНОГО ЗАПРОСА ВНУТРЬ ДРУГОГО
 
 
 
 
   В КОНЕЦ ГЛАВЫ 9,  МЫ ГОВОРИЛИ ЧТО ЗАПРОСЫ  могут  управлять  другими
 запросами.  В этой главе, вы узнаете как это делается ( большей частью
 ),  помещая запрос внутрь предиката другого запроса, и используя вывод
 внутреннего запроса в верном или неверном условии предиката. Вы сможе-
 те выяснить какие виды операторов могут использовать подзапросы и пос-
 мотреть  как  подзапросы работают со средствами SQL ,  такими как DIS-
 TINCT,  с составными функциями и выводимыми выражения.  Вы узнаете как
 использовать  подзапросы  с  предложением  HAVING и получите некоторые
 наставления как правильно использовать подзапросы.
 
 
             ========   КАК РАБОТАЕТ ПОДЗАПРОС?   =========
 
 
   С помощью SQL вы можете вкладывать запросы внутрь друга друга. Обыч-
 но, внутренний запрос генерирует значение которое проверяется в преди-
 кате внешнего запроса,  определяющего верно  оно  или  нет.  Например,
 предположим  что мы знаем имя продавца:  Motika,  но не знаем значение
 его поля snum, и хотим извлечь все порядки из таблицы Порядков. Имеет-
 ся один способ чтобы сделать это( вывод показывается в Рисунке 10.1 ):
 
 
     SELECT *
        FROM Orders
        WHERE snum =
            ( SELECT snum
                 FROM Salespeople
                 WHERE sname = 'Motika');
 
 
   Чтобы оценить внешний( основной ) запрос, SQL сначала должен оценить
 внутренний запрос ( или подзапрос ) внутри предложения WHERE. Он дела-
 ет это так как и должен делать  запрос  имеющий  единственную  цель  -
 отыскать через таблицу Продавцов все строки, где поле sname равно зна-
 чению Motika, и затем извлечь значения поля snum этих строк.
 
 
   Единственной найденной строкой естественно будет snum = 1004. Однако
 SQL, не просто выдает это значение, а помещает его в предикат основно-
 го запроса вместо самого подзапроса, так чтобы предиката прочитал что
 
 
        WHERE snum = 1004
 
 
                ===============  SQL Execution Log ==============
               |                                                 |
               | SELECT *                                        |
               | FROM  Orders                                    |
               | WHERE snum =                                    |
               | (SELECT snum                                    |
               | FROM Salespeople                                |
               | WHERE sname = 'Motika');                        |
               |=================================================|
               |   onum       amt      odate      cnum     snum  |
               |  -----     -------  ----------  -----    -----  |
               |   3002     1900.10  10/03/1990   2007     1004  |
               |                                                 |
                =================================================
 
 
 Рисунок 10.1:   Использование подзапроса
 
 
   Основной запрос  затем  выполняется как обычно с вышеупомянутыми ре-
 зультатами.  Конечно же,  подзапрос должен выбрать один и только  один
 столбец, а тип данных этого столбца должен совпадать с тем значением с
 которым он будет сравниваться в предикате.  Часто,  как показано выше,
 выбранное  поле  и  его  значение будут иметь одинаковые имена( в этом
 случае, snum ), но это необязательно.
   Конечно, если  бы  мы  уже знали номер продавца Motika,  мы могли бы
 просто напечатать
 
 
              WHERE snum = 1004
 
 
 и выполнять далее с подзапросом в целом, но это было бы не так универ-
 сально.  Это будет продолжать работать даже если номер Motika изменил-
 ся,  а, с помощью простого изменения имени в подзапросе, вы можете ис-
 пользовать его для чего угодно.
 
 
              ЗНАЧЕНИЯ, КОТОРЫЕ МОГУТ ВЫДАВАТЬ ПОДЗАПРОСЫ
 
 
   Скорее всего было бы удобнее,  чтобы наш подзапрос в предыдущем при-
 мере возвращал одно и только одно значение.
   Имея выбранным поле snum " WHERE city = "London" вместо "WHERE sname
 = 'Motika",  можно получить несколько различных  значений.  Это  может
 сделать уравнение в предикате основного запроса невозможным для оценки
 верности или неверности, и команда выдаст ошибку.
   При использовании подзапросов в предикатах основанных на реляционных
 операторах ( уравнениях или неравенствах,  как объяснено в Главе 4  ),
 вы  должны убедиться что использовали подзапрос который будет выдавать
 одну и только одну строку вывода.  Если вы используете подзапрос кото-
 рый  не выводит никаких значений вообще,  команда не потерпит неудачи;
 но основной запрос не выведет никаких значений.  Подзапросы которые не
 производят никакого вывода (или нулевой вывод) вынуждают рассматривать
 предикат ни как верный ни как неверный, а как неизвестный. Однако, не-
 известный  предикат имеет тот же самый эффект что и неверный:  никакие
 строки не выбираются основным запросом ( смотри Главу 5 для  подробной
 информации о неизвестном предикате ).
   Это плохая стратегия, чтобы делать что-нибудь подобное следующему:
 
 
                  SELECT *
                     FROM Orders
                     WHERE snum =
                       ( SELECT snum
                            FROM Salespeople
                            WHERE city = Barcelona );
 
 
   Поскольку мы  имеем только одного продавца в Barcelona - Rifkin,  то
 подзапрос будет выбирать одиночное значение snum и следовательно будет
 принят.  Но  это - только в данном случае.  Большинство SQL баз данных
 имеют многочисленых пользователей,  и если другой пользователь добавит
 нового  продавца из Barcelona в таблицу,  подзапрос выберет два значе-
 ния, и ваша команда потерпит неудачу.
 
 
                        DISTINCT С ПОДЗАПРОСАМИ
 
 
   Вы можете, в некоторых случаях, использовать DISTINCT чтобы вынудить
 подзапрос  генерировать  одиночное значение.  Предположим что мы хотим
 найти все порядки кредитований для тех продавцов  которые  обслуживают
 Hoffmanа ( cnum = 2001 ).
   Имеется один способ чтобы сделать это ( вывод показывается в Рисунке
 10.2 ):
 
 
             SELECT *
                FROM Orders
                WHERE snum =
                   ( SELECT DISTINCT snum
                        FROM Orders
                        WHERE cnum = 2001 );
 
 
                ===============  SQL Execution Log ==============
               |                                                 |
               | SELECT *                                        |
               | FROM  Orders                                    |
               | WHERE snum =                                    |
               | (SELECT DISTINCT snum                           |
               | FROM Orders                                     |
               | Where cnum = 2001);                             |
               | =============================================== |
               |   onum       amt      odate      cnum     snum  |
               |  -----   ---------  ---------   ------  ------- |
               |   3003      767.19  10/03/1990   2001     1001  |
               |   3008     4723.00  10/05/1990   2006     1001  |
               |   3011     9891.88  10/06/1990   2006     1001  |
                 ================================================
 
 
 Рисунок 10.2: Использование DISTINCT чтобы вынудить получение одного
                     значения из подзапроса
 
 
   Подзапрос установил что значение поля snum совпало с Hoffman - 1001,
 и затем основной запрос выделил все порядки с этим значением  snum  из
 таблицы Порядков( не разбирая,  относятся они к Hoffman или нет).  Так
 как каждый заказчик назначен к одному и только этому продавцу, мы зна-
 ем что каждая строка в таблице Порядков с данным значением cnum должна
 иметь такое же значение snum. Однако так как там может быть любое чис-
 ло  таких строк,  подзапрос мог бы вывести много ( хотя и идентичных )
 значений snum для данного поля cnum.  Аргумент DISTINCT  предотвращает
 это.  Если  наш  подзапрос возвратит более одного значения,  это будет
 указывать на ошибку в наших данных - хорошая вещь для знающих об этом.
 Альтернативный подход должен быть чтобы ссылаться к таблице Заказчиков
 а не к таблице Порядков в подзапросе.  Так как поле cnum - это первич-
 ный  ключ  таблицы Заказчика,  запрос выбирающий его должен произвести
 только одно значение.  Это рационально только если вы как пользователь
 имеете  доступ  к таблице Порядков но не к таблице Заказчиков.  В этом
 случае, вы можете использовать решение которое мы показали выше. ( SQL
 имеет механизмы которые определяют - кто имеет привилегии чтобы делать
 что-то в определенной таблице.  Это будет объясняться в Главе 22.) По-
 жалуйста  учтите,  что  методика используемая в предшествующем примере
 применима только когда вы знаете,  что два различных  поля  в  таблице
 должны всегда совпадать,  как в нашем случае. Эта ситуация не является
 типичной в реляционных базах данных, она являеться исключением из пра-
 вил.
 
 
             ПРЕДИКАТЫ С ПОДЗАПРОСАМИ ЯВЛЯЮТСЯ НЕОБРАТИМЫМИ
 
 
   Вы должны обратить внимание что предикаты включающие подзапросы, ис-
 пользуют выражение
 
 
    < скалярная форма > < оператор > < подзапрос >,       а не
    < подзапрос > < оператор > < скалярное выражение >    или,
    < подзапрос > < оператор > < подзапрос >.
 
 
   Другими словами, вы не должны записывать предыдущий пример так:
 
 
                    SELECT *
                       FROM Orders
                       WHERE ( SELECT DISTINCT snum
                            FROM Orders
                            WHERE cnum = 2001 )
                       = snum;
 
 
   В строгой  ANSI реализации,  это приведет к неудаче,  хотя некоторые
 программы и позволяют делать такие вещи.  ANSI также предохраняет  вас
 от появления обеих значений при сравнении, которые нужно вывести с по-
 мощью подзапроса.
 
 
             ИСПОЛЬЗОВАНИЕ АГРЕГАТНЫХ ФУНКЦИЙ В ПОДЗАПРОСАХ
 
 
   Один тип функций,  который автоматически может производить одиночное
 значение для любого числа строк, конечно же, - агрегатная функция.
   Любой запрос использующий одиночную функцию агрегата без предложения
 GROUP  BY будет выбирать одиночное значение для использования в основ-
 ном предикате.  Например,  вы хотите увидеть все порядки имеющие сумму
 приобретений  выше  средней на 4-е Октября ( вывод показан на Рисуноке
 10.3 ):
 
 
              SELECT *
                  FROM Orders
                  WHERE amt >
                     ( SELECT AVG (amt)
                          FROM Orders
                          WHERE odate = 10/04/1990 );
 
 
                ===============  SQL Execution Log ==============
               |                                                 |
               | SELECT *                                        |
               | FROM  Orders                                    |
               | WHERE amt >                                     |
               | (SELECT AVG (amt)                               |
               | FROM Orders                                     |
               | WHERE odate = 01/04/1990 );                     |
               | =============================================== |
               |   onum       amt      odate      cnum     snum  |
               |  -----    --------  ----------  -----    -----  |
               |   3002     1900.10  10/03/1990   2007     1004  |
               |   3005     2345.45  10/03/1990   2003     1002  |
               |   3006     1098.19  10/03/1990   2008     1007  |
               |   3009     1713.23  10/04/1990   2002     1003  |
               |   3008     4723.00  10/05/1990   2006     1001  |
               |   3010     1309.95  10/06/1990   2004     1002  |
               |   3011     9891.88  10/06/1990   2006     1001  |
                 ================================================
 
 
 Рисунок 10.3: Выбор всех сумм со значением выше средней на 10/04/1990
 
 
   Средняя сумма приобретений на 4 Октября - 1788.98 ( 1713.23 + 75.75)
 делится пополам,  что в целом равняется = 894.49. Все строки со значе-
 нием в поле amt выше этого - являются выбраными.
   Имейте ввиду что сгруппированные агрегатные функции, которые являют-
 ся  агрегатными  функциями  определенными в терминах предложения GROUP
 BY,  могут производить многочисленые значения.  Они, следовательно, не
 позволительны в подзапросах такого характера. Даже если GROUP BY и HA-
 VING используются таким способом,  что только одна группа выводится  с
 помощью подзапроса,  команда будет отклонена в принципе. Вы должны ис-
 пользовать одиночную агрегатную функцию с предложением WHERE что  уст-
 ранит нежелательные группы.  Например, следующий запрос который должен
 найти среднее значение комиссионных продавца в Лондоне -
 
 
                 SELECT AVG (comm)
                     FROM Salespeople
                     GROUP BY city
                     HAVlNG city = "London";
 
 
 не может использоваться в подзапросе! Во всяком случае это не лучший спо-
 соб формировать запрос. Другим способом может быть -
 
 
                  SELECT AVG (comm)
                     FROM Salespeople
                     WHERE city = "London";
 
 
 
 
                ИСПОЛЬЗОВАНИЕ ПОДЗАПРОСОВ КОТОРЫЕ ВЫДАЮТ
                   МНОГО СТРОК С ПОМОЩЬЮ ОПЕРАТОРА IN
 
 
   Вы можете использовать подзапросы  которые  производят  любое  число
 строк если вы используете специальный оператор IN ( операторы BETWEEN,
 LIKE,  и IS NULL не могут использоваться с подзапросами ). Как вы пом-
 ните, IN определяет набор значений, одно из которых должно совпадать с
 другим термином уравнения предиката в порядке, чтобы предикат был вер-
 ным.  Когда вы используете IN с подзапросом, SQL просто формирует этот
 набор из вывода подзапроса.  Мы можем,  следовательно, использовать IN
 чтобы выполнить такой же подзапрос который не будет работать с реляци-
 онным оператором, и найти все атрибуты таблицы Порядков для продавца в
 Лондоне ( вывод показывается в Рисунке 10.4 ):
 
 
             SELECT *
                 FROM Orders
                 WHERE snum IN
                      ( SELECT snum
                          FROM Salespeople
                          WHERE city = "LONDON" );
 
 
                ===============  SQL Execution Log ==============
               |                                                 |
               | SELECT *                                        |
               | FROM  Orders                                    |
               | WHERE snum IN                                   |
               | (SELECT snum                                    |
               | FROM Salespeople                                |
               | WHERE city = 'London');                         |
               | =============================================== |
               |   onum       amt      odate      cnum     snum  |
               |  -----    --------  ----------  -----   ------  |
               |   3003      767.19  10/03/1990   2001     1001  |
               |   3002     1900.10  10/03/1990   2007     1004  |
               |   3006     1098.19  10/03/1990   2008     1007  |
               |   3008     4723.00  10/05/1990   2006     1001  |
               |   3011     9891.88  10/06/1990   2006     1001  |
                 ================================================
 
 
 Рисунок 10. 4:  Использование подзапроса с IN
 
 
   В ситуации подобно этой,  подзапрос - более прост  для  пользователя
 чтобы  понимать  его и более прост для компьютера чтобы его выполнить,
 чем если бы Вы использовали обьединение:
 
 
        SELECT onum, amt, odate, cnum, Orders.snum
           FROM Orders, Salespeople
           WHERE Orders.snum = Salespeople.snum
               AND Salespeople.city = "London";
 
 
   Хотя это и произведет тот же самый вывод что и в примере с подзапро-
 сом, SQL должен будет просмотреть каждую возможную комбинацию строк из
 двух таблиц и проверить их снова по составному предикату.  Проще и эф-
 фективнее извлекать из таблицы Продавцов значения поля snum где city =
 "London",  и затем искать эти значения в таблице Порядков, как это де-
 лается в варианте с подзапросом. Внутренний запрос дает нам snums=1001
 и snum=1004.  Внешний запрос, затем, дает нам строки из таблицы Поряд-
 ков где эти поля snum найдены.
   Строго говоря,  быстрее или нет работает вариант подзапроса, практи-
 чески зависит от реализации - в какой программе  вы  это  используете.
 Эта часть вашей программы называемой - оптимизатор, пытается найти на-
 иболее эффективный способ выполнения ваших запросов.
   Хороший оптимизатор во всяком случае преобразует вариант обьединения
 в подзапрос, но нет достаточно простого способа для вас чтобы выяснить
 выполнено это или нет. Лучше сохранить ваши запросы в памяти чем пола-
 гаться полностью на оптимизатор.
   Конечно вы можете также использовать оператор IN, даже когда вы уве-
 рены что подзапрос произведет одиночное значение. В любой ситуации где
 вы  можете использовать реляционный оператор сравнения (=),  вы можете
 использовать IN. В отличие от реляционных операторов, IN не может зас-
 тавить команду потерпеть неудачу если больше чем одно значение выбрано
 подзапросом.  Это может быть или преимуществом или недостатком.  Вы не
 увидите  непосредственно вывода из подзапросов;  если вы полагаете что
 подзапрос собирается произвести только одно значение,  а он производит
 различные.  Вы не сможете объяснить различия в выводе основного запро-
 са. Например, рассмотрим команду, которая похожа на предыдущую:
 
 
               SELECT onum, amt, odate
                   FROM Orders
                   WHERE snum =
                       ( SELECT  snum
                       FROM Orders
                       WHERE cnum = 2001 );
 
 
   Вы можете  устранить потребность в DISTINCT используя IN вместо (=),
 подобно этому:
 
 
               SELECT onum, amt, odate
                   FROM Orders
                   WHERE snum IN
                       ( SELECT snum
                       FROM Orders
                       WHERE cnum = 2001 );
 
 
   Что случится  если  есть ошибка и один из порядков был акредитован к
 различным продавцам?  Версия использующая IN будет давать вам все  по-
 рядки для обоих продавцов.  Нет никакого очевидного способа наблюдения
 за ошибкой,  и поэтому сгенерированные отчеты или решения сделанные на
 основе этого запроса не будут содержать ошибки. Вариант использующий (
 = ) , просто потерпит неудачу.
   Это, по крайней мере, позволило вам узнать что имеется такая пробле-
 ма.  Вы должны затем выполнять поиск неисправности, выполнив этот под-
 запрос отдельно и наблюдая значения которые он производит.
   В принципе,  если вы знаете что подзапрос должен( по логике) вывести
 только одно значение,  вы должны использовать = . IN является подходя-
 щим,  если запрос может ограниченно производить одно или более  значе-
 ний,  независимо от того ожидаете вы их или нет. Предположим, мы хотим
 знать комиссионные всех продавцов обслуживающих заказчиков в Лондоне:
 
 
           SELECT comm
               FROM Salespeople
               WHERE snum IN
                 ( SELECT snum
                     FROM Customers
                     WHERE city = "London" );
 
 
   Выводимыми для этого запроса,  показанного в Рисунке 10.5,  являются
 значения комиссионных продавца Peel ( snum =  1001  ),  который  имеет
 обоих заказчиков в Лондоне. Это - только для данного случая. Нет ника-
 кой причины чтобы некоторые заказчики в Лондоне не могли быть назначе-
 ными  к кому-то еще.  Следовательно,  IN - это наиболее логичная форма
 чтобы использовать ее в запросе.
 
 
                    ===============  SQL Execution Log ==============
                   |                                                 |
                   | SELECT comm                                     |
                   | FROM  Salespeople                               |
                   | WHERE snum IN                                   |
                   | (SELECT snum                                    |
                   | FROM Customers                                  |
                   | WHERE city = 'London');                         |
                   | =============================================== |
                   |    comm                                         |
                   |  -------                                        |
                   |    0.12                                         |
                   |                                                 |
                   |                                                 |
                     ================================================
 
 
 
 
 Рисунок 10.5  Использование IN с подзапросом для вывода одного значения
 
 
   Между прочим,  префикс таблицы для поля city необязателен в предыду-
 щем примере,  несмотря на возможную неоднозначность между полями  city
 таблицы Заказчика и таблицы Продавцов.
   SQL всегда ищет первое поле в  таблице  обозначенной  в  предложении
 FROM  текущего  подзапроса.  Если поле с данным именем там не найдено,
 проверяются внешние запросы.  В вышеупомянутом примере, "city" в пред-
 ложении  WHERE  означает что имеется ссылка к Customer.city( поле city
 таблицы Заказчиков).  Так как таблица Заказчиков указана в предложении
 FROM текущего запроса, SQL предполагает что это - правильно. Это пред-
 положение может быть отменено  полным  именем  таблицы  или  префиксом
 псевдонима, о которых мы поговорим позже когда будем говорить об соот-
 несенных подзапросах. Если возможен беспорядок, конечно же, лучше все-
 го использовать префиксы.
 
 
                 ПОДЗАПРОСЫ ВЫБИРАЮТ ОДИНОЧНЫЕ СТОЛБЦЫ
 
 
   Смысл всех подзапросов обсужденных в этой главе тот, что все они вы-
 бирают одиночный столбец.  Это обязательно,  поскольку выбранный вывод
 сравнивается с одиночным значением.  Подтверждением этому то,  что SE-
 LECT * не может использоваться в  подзапросе.  Имеется  исключение  из
 этого,  когда подзапросы используются с оператором EXISTS,  который мы
 будем представлять в Главе 12.
 
 
                 ИСПОЛЬЗОВАНИЕ ВЫРАЖЕНИЙ В ПОДЗАПРОСАХ
 
 
   Вы можете использовать выражение основанное на столбце,  а не просто
 сам столбец, в предложении SELECT подзапроса. Это может быть выполнено
 или с помощью реляционных операторов или  с  IN.  Например,  следующий
 запрос использует реляционный оператор = ( вывод показывается в Рисун-
 ке 10.6 ):
 
 
              SELECT *
                 FROM Customers
                 WHERE cnum =
                     ( SELECT snum + 1000
                          FROM Salespeople
                          WHERE sname = Serres );
 
 
   Он находит всех заказчиков чье значение поля cnum равное 1000,  выше
 поля snum Serres.  Мы предполагаем что столбец sname не имеет  никаких
 двойных значений ( это может быть предписано или UNIQUE INDEX,  обсуж-
 даемым в Главе 17,  или ограничением UNIQUE, обсуждаемым в Главе 18 );
 иначе
                ===============  SQL Execution Log ============
               |                                               |
               | SELECT *                                      |
               | FROM  Customers                               |
               | WHERE cnum =                                  |
               | (SELECT snum + 1000                           |
               | WHERE Salespeople                             |
               | WHERE sname = 'Serres'                        |
               | ============================================= |
               |   cnum     cname     city    rating    snum   |
               |  -----    --------   ----    ------   -----   |
               |   2002    Giovanni   Rome       200    1003   |
                 =============================================
 
 
 Рисунок 10.6: Использование подзапроса с выражением
 
 
 подзапрос может произвести многочисленые значения.  Когда поля snum  и
 сnum  не  имеют  такого простого функционального значения как например
 первичный ключ , что не всегда хорошо, запрос типа вышеупомянутого не-
 вероятно полезен.
 
 
 
 
                    ПОДЗАПРОСЫ В ПРЕДЛОЖЕНИИ HAVING
 
 
   Вы можете также использовать подзапросы внутри  предложения  HAVING.
 Эти  подзапросы могут использовать свои собственные агрегатные функции
 если они не производят многочисленых значений или  использовать  GROUP
 BY или HAVING.  Следующий запрос является этому примером ( вывод пока-
 зывается в Рисунке 10.7 ):
 
 
         SELECT rating, COUNT ( DISTINCT cnum )
             FROM Customers
             GROUP BY rating
             HAVING rating >
                 ( SELECT AVG (rating)
                      FROM Customers
                      WHERE city = " San Jose';
 
 
                ===============  SQL Execution Log ============
               |                                               |
               | SELECT rating,count (DISTINCT cnum)           |
               | FROM  Customers                               |
               | GROUP BY rating                               |
               | HAVING rating >                               |
               | (SELECT AVG (rating)snum + 1000               |
               | FROM Custimers                                |
               | WHERE city = 'San Jose'                       |
               |============================================   |
               |  rating                                       |
               | --------    --------                          |
               |   200             2                           |
                ================================================
 
 
 Рисунок 10.7: Нахождение заказчиков с оценкой выше среднего в San Jose
 
 
   Эта команда  подсчитывает  заказчиков с оценками выше среднего в San
 Jose.  Так как имеются другие оценки отличные от 300,  они должны быть
 выведены с числом номеров заказчиков которые имели эту оценку.
 
 
               ===============   РЕЗЮМЕ   ===============
 
 
   Теперь вы используете запросы в иерархической манере. Вы видели, что
 использование результата одного запроса для управления другим,  расши-
 ряет возможности позволяющие выполнить большее количество функций.  Вы
 теперь понимаете как использовать подзапросы с реляционными оператора-
 ми  также как и со специальным оператором IN,  или в предложении WHERE
 или в предложении HAVING внешнего запроса.
   В следующих  главах,  мы  будем разрабатывать подзапросы.  Сначала в
 Главе 11,  мы обсудим другой вид подзапроса,  который выполняется  от-
 дельно для каждой строки таблицы вызываемой во внешнем запросе. Затем,
 в Главе 12 и 13,  мы представим вас нескольким специальным  операторам
 которые функционируют на всех подзапросах,  как это делает IN, за иск-
 лючением когда эти операторы могут использоваться только  в  подзапро-
 сах.
 
 
               *************** РАБОТА С SQL *************
 
 
 1. Напишите  запрос,  который  бы  использовал подзапрос для получения
    всех порядков для заказчика с именем Cisneros.  Предположим, что вы
    не знаете номера этого заказчика, указываемого в поле cnum.
 2. Напишите запрос который вывел бы имена и оценки всех заказчиков ко-
    торые имеют усредненые порядки.
 3. Напишите запрос который бы выбрал общую сумму всех  приобретений  в
    порядках  для  каждого продавца,  у которого эта общая сумма больше
    чем сумма наибольшего порядка в таблице.
 
 
 ( См. Приложение A для ответов. )
 
 
 
Перейти к оглавлению
Hosted by uCoz