Базы данных: SQL
Понимание SQL. Глава 6. Обобщение данных с помощью агрегатных функций
Источник:
В этой главе, вы перейдете от простого
использования запросов к извлечению значений из базы данных и
определению, как вы можете использовать эти значения чтобы
получить из них информацию. Это делается с помощью агрегатных
или общих функций которые берут группы значений из пол и
сводят их до одиночного значения. Вы узнаете как использовать
эти функции, как определить группы значений к которым они
будут применяться, и как определить какие группы выбираются
для вывода. Вы будете также видеть при каких условиях вы
сможете объединить значения пол с этой полученной информацией
в одиночном запросе.
ЧТО ТАКОЕ АГРЕГАТНЫЕ ФУНКЦИИ ?Запросы могут
производить обобщенное групповое значение полей точно также
как и значение одного пол. Это делает с помощью агрегатных
функций. Агрегатные функции производят одиночное значение для
всей группы таблицы. Имеется список этих функций:
*
COUNT - производит номера строк или не-NULL значения
полей которые выбрал запрос.
* SUM - производит
арифметическую сумму всех выбранных значений данного пол.
* AVG - производит усреднение всех выбранных
значений данного пол.
* MAX - производит
наибольшее из всех выбранных значений данного пол.
*
MIN - производит наименьшее из всех выбранных значений
данного пол.
КАК ИСПОЛЬЗОВАТЬ АГРЕГАТНЫЕ ФУНКЦИИ ?Агрегатные
функции используются подобно именам полей в предложении 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.
РЕЗЮМЕТеперь вы используете запросы несколько
по-другому. Способность получать, а не просто размещать
значения, очень мощна. Это означает что вы не обязательно
должны следить за определенной информацией если вы можете
сформулировать запрос так чтобы ее получить. Запрос будет
давать вам поминутные результаты, в то врем как таблица общего
или среднего значений будет хороша только некоторое врем после
ее модификации. Это не должно наводить на мысль, что
агрегатные функции могут полностью вытеснить потребность в
отслеживании информации такой например как эта. Вы можете
применять эти агрегаты для групп значений определенных
предложением GROUP BY. Эти группы имеют значение поля в целом,
и могут постоянно находиться внутри других групп которые имеют
значение поля в целом. В то же время, предикаты еще
используются чтобы определять какие строки агрегатной функции
применяются. Объединенные вместе, эти особенности делают
возможным, производить агрегаты основанные на сильно
определенных подмножествах значений в поле. Затем вы можете
определять другое условие для исключения определенных
результатов групп с предложением HAVING. Теперь , когда вы
стали знатоком большого количества того как запрос производит
значения, мы покажем вам, в Главе 7, некоторые вещи которые вы
можете делать со значениями которые он производит.
РАБОТА С SQL
1. Напишите запрос который
сосчитал бы все суммы приобретений на 3 Октября.
2.
Напишите запрос который сосчитал бы число различных не-NULL
значений пол city в таблице Заказчиков.
3. Напишите
запрос который выбрал бы наименьшую сумму для каждого
заказчика.
4. Напишите запрос который бы выбирал
заказчиков в алфавитном порядке, чьи имена начинаются с буквы
G.
5. Напишите запрос который выбрал бы высшую оценку
в каждом городе.
6. Напишите запрос который сосчитал
бы число заказчиков регистрирующих каждый день свои порядки.
(Если продавец имел более одного порядка в данный день, он
должен учитываться только один раз.)
(См. Приложение
A для ответов.)
Назад | Далее
При перепечатке любого материала
с сайта, видимая ссылка на источник www.warayg.narod.ru
и все имена, ссылки авторов обязательны.
© 2005
|