В ЭТОЙ ГЛАВЕ, ВЫ ПЕРЕЙДЕТЕ ОТ ПРОСТОГО
использования запросов к извлечению значений из базы данных и
определению, как вы можете использовать эти значения чтобы получить
из них информацию. Это делается с помощью агрегатных или общих
функций которые берут группы значений из пол и сводят их до
одиночного значения. Вы узнаете как использовать эти функции, как
определить группы значений к которым они будут применяться, и как
определить какие группы выбираются для вывода. Вы будете также
видеть при каких условиях вы сможете объединить значения пол с
этой полученной информацией в одиночном запросе.
КАК ИСПОЛЬЗОВАТЬ АГРЕГАТНЫЕ ФУНКЦИИ ?
Агрегатные функции используются подобно именам полей в предложении
SELECT запроса, но с одним исключением, они берут имена пол как
аргументы. Только числовые пол могут использоваться с SUM и AVG.
С COUNT, MAX, и MIN, могут использоваться и числовые или
символьные пол. Когда они используются с символьными полями,
MAX и MIN будут транслировать их в эквивалент ASCII, который
должен сообщать, что MIN будет означать первое, а MAX последнее
значение в алфавитном порядке( выдача алфавитного упорядочения
обсуждается более подробно в Главе 4 ).
Чтобы найти SUM всех наших покупок в таблицы Порядков, мы можем
ввести следующий запрос, с его выводом в Рисунке 6.1:
SELECT SUM ((amt))
FROM Orders;
=============== SQL Execution Log ============
| |
| SELECT SUM (amt) |
| FROM Orders; |
| ==============================================|
| |
| ------- |
| 26658.4 |
| |
| |
===============================================
Рисунок 6.1: Выбор суммы
Это конечно, отличается от выбора пол при котором возвращается
одиночное значение, независимо от того сколько строк находится в
таблице. Из-за этого, агрегатные функции и пол не могут выбираться
одновременно, пока предложение GROUP BY (описанное далее)
не будет использовано.
Нахождение усредненной суммы - это похожа операция ( вывод
следующего запроса показывается в Рисунке 6.2 ):
SELECT AVG (amt)
FROM Orders;
=============== SQL Execution Log ============
| |
| SELECT AVG (amt) |
| FROM Orders; |
| ==============================================|
| |
| ------- |
| 2665.84 |
| |
| |
===============================================
Рисунок 6.2: Выбор среднего
СПЕЦИАЛЬНЫЕ АТРИБУТЫ COUNT
Функция COUNT несколько отличается от всех. Она считает число
значений в данном столбце, или число строк в таблице. Когда она
считает значения столбца, она используется с DISTINCT чтобы
производить счет чисел различных значений в данном поле. Мы могли
бы использовать ее, например, чтобы сосчитать номера продавцов в
настоящее врем описанных в таблице Порядков ( вывод показывается
в Рисунке 6.3 ):
SELECT COUNT ( DISTINCT snum )
FROM Orders;
ИСПОЛЬЗОВАНИЕ DISTINCT
Обратите внимание в вышеупомянутом примере, что DISTINCT,
сопровождаемый именем пол с которым он применяется, помещен
в круглые скобки, но не сразу после SELECT, как раньше.
Этого использования DISTINCT с COUNT применяемого к
индивидуальным столбцам, требует стандарт ANSI, но большое
количество программ не предъявляют к ним такого требования.
=============== SQL Execution Log ============
| |
| SELECT COUNT (DISTINCT snum) |
| FROM Orders; |
| ==============================================|
| |
| ------- |
| 5 |
| |
| |
===============================================
Рисунок 6.3: Подсчет значений пол
Вы можете выбирать многочисленные счета( COUNT ) из полей с помощью
DISTINCT в одиночном запросе который, как мы видели в
Главе 3, не
выполнялись когда вы выбирали строки с помощью DISTINCT.
DISTINCT может использоваться таким образом, с любой функцией агрегата,
но наиболее часто он используется с COUNT. С MAX и MIN, это просто
не будет иметь никакого эффекта, а SUM и AVG, вы обычно применяете
для включения повторяемых значений, так как они законно эффективнее
общих и средних значений всех столбцов.
ИСПОЛЬЗОВАНИЕ COUNT СО СТРОКАМИ, А НЕ ЗНАЧЕНИЯМИ
Чтобы подсчитать общее число строк в таблице, используйте функцию
COUNT со звездочкой вместо имени пол, как например в следующем
примере, вывод из которого показан на Рисунке 6.4:
SELECT COUNT (*)
FROM Customers
COUNT со звездочкой включает и NULL и дубликаты, по этой причине
DISTINCT не может быть использован. DISTINCT может производить
более высокие номера чем COUNT особого пол, который удаляет все
=============== SQL Execution Log ============
| |
| SELECT COUNT (*) |
| FROM Customers; |
| ==============================================|
| |
| ------- |
| 7 |
| |
| |
===============================================
Рисунок 6. 4: Подсчет строк вместо значений
строки, имеющие избыточные или NULL данные в этом поле.
DISTINCT не применим c COUNT (*), потому, что он не имеет никакого
действия в хорошо разработанной и поддерживаемой базе данных.
В такой базе данных, не должно быть ни таких строк, которые бы являлись
полностью пустыми, ни дубликатов ( первые не содержат никаких
данных, а последние полностью избыточны ). Если, с другой стороны,
все таки имеются полностью пустые или избыточные строки, вы
вероятно не захотите чтобы COUNT скрыл от вас эту информацию.
ВКЛЮЧЕНИЕ ДУБЛИКАТОВ В АГРЕГАТНЫЕ ФУНКЦИИ
Агрегатные функции могут также ( в большинстве реализаций )
использовать аргумент ALL, который помещается перед именем пол,
подобно DISTINCT, но означает противоположное: - включать дубликаты.
ANSI технически не позволяет этого для COUNT, но многие реализации
ослабляют это ограничение.
Различи между ALL и * когда они используются с COUNT -
- ALL использует имя_поля как аргумент.
- ALL не может подсчитать значения NULL.
Пока * является единственным аргументом который включает NULL
значения, и он используется только с COUNT; функции отличные от
COUNT игнорируют значения NULL в любом случае. Следующая
команда подсчитает(COUNT) число не-NULL значений в поле rating
в таблице Заказчиков ( включая повторения ):
SELECT COUNT ( ALL rating )
FROM Customers;
АГРЕГАТЫ ПОСТРОЕННЫЕ НА СКАЛЯРНОМ ВЫРАЖЕНИИ
До этого, вы использовали агрегатные функции с одиночными полями как
аргументами. Вы можете также использовать агрегатные функции с
аргументами которые состоят из скалярных выражений включающих одно
или более полей. ( Если вы это делаете, DISTINCT не разрешается. )
Предположим, что таблица Порядков имеет еще один столбец который
хранит предыдущий неуплаченный баланс (поле blnc) для каждого заказчика.
Вы должны найти этот текущий баланс, добавлением суммы приобретений
к предыдущему балансу.
Вы можете найти наибольший неуплаченный баланс следующим образом:
SELECT MAX ( blnc + (amt) )
FROM Orders;
Для каждой строки таблицы, этот запрос будет складывать blnc и amt
для этого заказчика и выбирать самое большое значение которое он
найдет. Конечно, пока заказчики могут иметь многочисленные порядки,
их неуплаченный баланс оценивается отдельно для каждого порядка.
Возможно, порядок с более поздней датой будет иметь самый большой
неуплаченный баланс. Иначе, старый баланс должен быть выбран как в
запросе выше.
Фактически, имеются большое количество ситуаций в SQL где вы
можете использовать скалярные выражения с полями или вместо полей,
как вы увидите это в Главе 7.
ПРЕДЛОЖЕНИЕ GROUP BY
Предложение GROUP BY позволяет вам определять подмножество
значений в особом поле в терминах другого пол, и применять функцию
агрегата к подмножеству. Это дает вам возможность объединять пол и
агрегатные функции в едином предложении SELECT.
Например, предположим что вы хотите найти наибольшую сумму приобретений
полученную каждым продавцом. Вы можете сделать раздельный
запрос для каждого из них, выбрав MAX (amt) из таблицы Порядков для
каждого значения пол snum. GROUP BY, однако, позволит Вам поместить их
все в одну команду:
SELECT snum, MAX (amt)
FROM Orders
GROUP BY snum;
Вывод для этого запроса показывается в Рисунке 6.5.
=============== SQL Execution Log ==============
| |
| SELECT snum, MAX (amt) |
| FROM Orders |
| GROUP BY snum; |
| =============================================== |
| snum |
| ------ -------- |
| 1001 767.19 |
| 1002 1713.23 |
| 1003 75.75 |
| 1014 1309.95 |
| 1007 1098.16 |
| |
================================================
Рисунок 6.5: Нахождение максимальной суммы продажи у каждого продавца
GROUP BY применяет агрегатные функции независимо от серий групп
которые определяются с помощью значения поля в целом. В этом случае,
каждая группа состоит из всех строк с тем же самым значением пол snum,
и MAX функция применяется отдельно для каждой такой группы. Это значение
пол, к которому применяется GROUP BY, имеет, по определению, только одно
значение на группу вывода, также как это делает агрегатная функция.
Результатом является совместимость которая позволяет агрегатам и полям
объединяться таким образом.
Вы можете также использовать GROUP BY с многочисленными полями.
Совершенству вышеупомянутый пример далее, предположим что вы
хотите увидеть наибольшую сумму приобретений получаемую каждым
продавцом каждый день. Чтобы сделать это, вы должны сгруппировать
таблицу Порядков по датам продавцов, и применить функцию MAX к
каждой такой группе, подобно этому:
SELECT snum, odate, MAX ((amt))
FROM Orders
GROUP BY snum, odate;
Вывод для этого запроса показывается в Рисунке 6.6.
=============== SQL Execution Log ==============
| |
| SELECT snum, odate, MAX (amt) |
| FROM Orders |
| GROUP BY snum, odate; |
| =============================================== |
| snum odate |
| ------ ---------- -------- |
| 1001 10/03/1990 767.19 |
| 1001 10/05/1990 4723.00 |
| 1001 10/06/1990 9891.88 |
| 1002 10/03/1990 5160.45 |
| 1002 10/04/1990 75.75 |
| 1002 10/06/1990 1309.95 |
| 1003 10/04/1990 1713.23 |
| 1014 10/03/1990 1900.10 |
| 1007 10/03/1990 1098.16 |
| |
================================================
Рисунок 6.6: Нахождение наибольшей суммы приобретений на каждый день
Конечно же, пустые группы, в дни когда текущий продавец не имел порядков,
не будут показаны в выводе.
ПРЕДЛОЖЕНИЕ HAVING
Предположим, что в предыдущем примере, вы хотели бы увидеть только
максимальную сумму приобретений значение которой выше $3000.00. Вы
не сможете использовать агрегатную функцию в предложении WHERE
( если вы не используете подзапрос, описанный позже ), потому что
предикаты оцениваются в терминах одиночной строки, а агрегатные
функции оцениваются в терминах групп строк. Это означает что вы не
сможете сделать что-нибудь подобно следующему:
SELECT snum, odate, MAX (amt)
FROM Oreders
WHERE MAX ((amt)) > 3000.00
GROUP BY snum, odate;
Это будет отклонением от строгой интерпретации ANSI. Чтобы увидеть
максимальную стоимость приобретений свыше $3000.00, вы можете
использовать предложение HAVING.
Предложение HAVING определяет критерии используемые чтобы удалять
определенные группы из вывода, точно также как предложение WHERE
делает это для индивидуальных строк.
Правильной командой будет следующая:
SELECT snum, odate, MAX ((amt))
FROM Orders
GROUP BY snum, odate
HAVING MAX ((amt)) > 3000.00;
Вывод для этого запроса показывается в Рисунке 6. 7.
=============== SQL Execution Log ==============
| |
| SELECT snum, odate, MAX (amt) |
| FROM Orders |
| GROUP BY snum, odate |
| HAVING MAX (amt) > 3000.00; |
| =============================================== |
| snum odate |
| ------ ---------- -------- |
| 1001 10/05/1990 4723.00 |
| 1001 10/06/1990 9891.88 |
| 1002 10/03/1990 5160.45 |
| |
================================================
Рисунок 6. 7: Удаление групп агрегатных значений
Аргументы в предложении HAVING следуют тем же самым правилам
что и в предложении SELECT, состоящей из команд использующих
GROUP BY. Они должны иметь одно значение на группу вывода.
Следующая команда будет запрещена:
SELECT snum, MAX (amt)
FROM Orders
GROUP BY snum
HAVING odate = 10/03/1988;
Поле оdate не может быть вызвано предложением HAVING, потому что
оно может иметь ( и действительно имеет ) больше чем одно значение
на группу вывода. Чтобы избегать такой ситуации, предложение HAVING
должно ссылаться только на агрегаты и поля выбранные GROUP BY.
Имеется правильный способ сделать вышеупомянутый запрос( вывод
показывается в Рисунке 6.8 ):
SELECT snum, MAX (amt)
FROM Orders
WHEREodate = 10/03/1990
GROUP BY snum;
=============== SQL Execution Log ==============
| |
| SELECT snum, odate, MAX (amt) |
| FROM Orders |
| GROUP BY snum, odate; |
| =============================================== |
| snum |
| ------ -------- |
| 1001 767.19 |
| 1002 5160.45 |
| 1014 1900.10 |
| 1007 1098.16 |
| |
================================================
Рисунок 6.8: Максимальное значение суммы приобретений у каждого
продавца на 3 Октября
Поскольку пол odate нет, не может быть и выбранных полей, значение
этих данных меньше чем в некоторых других примерах.
Вывод должен вероятно включать что-нибудь такое что говорит -
" это - самые большие порядки на 3 Октября." В Главе 7,
мы покажем как вставлять текст в ваш вывод.
Как и говорилось ранее, HAVING может использовать только аргументы
которые имеют одно значение на группу вывода. Практически, ссылки
на агрегатные функции - наиболее общие, но и пол выбранные с помощью
GROUP BY также допустимы. Например, мы хотим увидеть наибольшие
порядки для Serres и Rifkin:
SELECT snum, MAX (amt)
FROM Orders
GROUP BY snum
HAVING snum B (1002,1007);
Вывод для этого запроса показывается в Рисунке 6.9.
=============== SQL Execution Log ==============
| |
| SELECT snum, MAX (amt) |
| FROM Orders |
| GROUP BY snum |
| HAVING snum IN ( 1002, 1007 ); |
| =============================================== |
| snum |
| ------ -------- |
| 1002 5160.45 |
| 1007 1098.16 |
| |
================================================
Рисунок 6. 9: Использование HAVING с GROUP BY полями
НЕ ДЕЛАЙТЕ ВЛОЖЕННЫХ АГРЕГАТОВ
В строгой интерпретации ANSI SQL, вы не можете использовать агрегат
агрегата. Предположим что вы хотите выяснить, в какой день имелась
наибольшая сумма приобретений. Если вы попробуете сделать это,
SELECT odate, MAX ( SUM (amt) )
FROM Orders
GROUP BY odate;
то ваша команда будет вероятно отклонена. ( Некоторые реализации не
предписывают этого ограничения, которое является выгодным, потому
что вложенные агрегаты могут быть очень полезны, даже если они и
несколько проблематичны.) В вышеупомянутой команде, например,
SUM должен применяться к каждой группе пол odate, а MAX ко
всем группам, производящим одиночное значение для всех групп.
Однако предложение GROUP BY подразумевает что должна иметься
одна строка вывода для каждой группы пол odate.