Группировка и агрегаты.

Агрегатные функции:

Max() Возвращает максимальное значение из набора.

Min() Возвращает минимальное значение из набора.

Avg() Возвращает среднее значение набора.

Sum() Возвращает сумму значений из набора.

Count() Возвращает количество значений в наборе.

SELECT MAX(price),

MIN(price),

AVG(price),

SUM(price),

COUNT(*)

FROM realization;

Мало кого заинтересуют необработанные данные; тем, кто занимается анализом, потребуются обработанные данные, приведенные к виду, наиболее соответствующему их нуждам.

Необходимо понять, что такое группировка:

ФИО ТОВАР КОЛИЧЕСТВО ЦЕНА
Иванов принтер
Петров мышь
Иванов принтер
Иванов мышь
Петров принтер
Петров принтер
Лосев Ноутбук
Лосев ноутбук
Иванов мышь

 

Группировка может быть по какому-либо столбцу. Например, группировка по ФИО будет выглядеть следующим образом:

ФИО ТОВАР КОЛИЧЕСТВО ЦЕНА
Иванов   ИТОГ  
Иванов принтер
Иванов принтер
Иванов мышь
Иванов мышь
Петров   ИТОГ  
Петров мышь
Петров принтер
Петров принтер
Лосев   ИТОГ  
Лосев Ноутбук
Лосев ноутбук

 

1. Это самая простая группировка:

SELECT customer, SUM(count*price)

FROM realization

GROUP BY customer;

Этот запрос выявит все группировки по клиентам и посчитает итог по каждой группировке. В данном случае итог – это сумма вычисляемого столбца count*price.

Здесь добавлен новый блок GROUP BY.

2. В некоторых случаях может понадобиться сформировать группы, охватывающие более одного столбца.

SELECT customer, product, SUM(count*price)

FROM realization

GROUP BY customer, product;

3. Если мы хотим увидеть не только окончательные итоги, но и промежуточные итоги, то необходимо использовать предложение with rollup в блоке group by:

SELECT customer, product, SUM(count*price)

FROM realization

GROUP BY customer, product with rollup;

4. При группировке данных тоже можно применять условия фильтрации к данным после

формирования групп. Этот тип условий фильтрации должен располагаться в блоке having.

SELECT customer, SUM(count*price)

FROM realization

GROUP BY customer

HAVING SUM(count*price)>5000;

Фильтр WHERE воздействует на данные до группировки, а HAVING– после создания групп. В блок having можно включить агрегатные функции, не перечисленные в блоке select, в блок WHERE нельзя включать агрегатные функции.


 

Связи между таблицами.

До сих пор мы работали с одной таблицей. Теперь посмотрим, как получать данные из нескольких таблиц. Большинство запросов все же обращены к двум, трем или даже более таблицам.

Чтобы получить информацию из нескольких таблиц их необходимо соединить. Соединений бывает несколько:

1. Внутреннее соединение

2. Внешнее соединение

3. Левое соединение

4. Правое соединение

5. Кросс-соединение

Для иллюстрации этих соединений рассмотрим две таблицы:

СОТРУДНИКИ
ID ФИО ID_Отдел
Иванов
Петров
Сидоров
Лосев
Чуркин
6 Лисин 5
Конев
Попов
Иваненко
Туманов

 

 

Отделы
ID Наименование
Бухгалтерия
Снабжение
Продажи
Маркетинг

 

 

Обратите внимание, что есть отдел маркетинга, в таблице Отделы, в этом отделе никто не работает и есть некто Лисин, который работает в отделе, который не существует.

Внутреннее соединение получит у нас данные из таблиц, но только те, для которых есть связные данные в каждой таблице. Запрос будет выглядеть следующим образом:

SELECT *

FROM сотрудники INNER JOIN отделы

ON Сотрудники.ID_Отдел = Отделы.ID;

 

Левое соединение. Левой таблицей считается таблица, которая в запросе встречается первая, вторая таблица будет правой. При левом соединении мы получим все записи из левой таблицы, независимо от того есть ли соответствующая запись в правой таблице. Если соответствующей записи не будет, мы получим в результате NULL значение. Запрос будет выглядеть следующим образом:

SELECT *

FROM сотрудники LEFT JOIN отделы

ON Сотрудники.ID_Отдел = Отделы.ID;

 

Правое соединение аналогично левому, только будут выведены все записи из правой таблицы.

Запрос будет выглядеть следующим образом:

SELECT *

FROM сотрудники RIGHT JOIN отделы

ON Сотрудники.ID_Отдел = Отделы.ID;

 

Внешнее соединение. Получим все записи из всех таблиц, и по возможности соответствующие записи из связных таблиц. Запрос будет выглядеть следующим образом:

SELECT *

FROM сотрудники FULL OUTER JOIN отделы

ON Сотрудники.ID_Отдел = Отделы.ID;

 

Кросс объединение. Соответствует операции произведения в реляционной алгебре. Условие связи отсутствует и мы получим всевозможные соответствия между двумя таблицами.

SELECT *

FROM сотрудники CROSS JOIN отделы ;

 

Рассмотрим пример.

Схема будет следующая:

Запрос для схемы:

-- MySQL Workbench Forward Engineering

SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0;

SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0;

SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='TRADITIONAL,ALLOW_INVALID_DATES';

-- Schema Lessons

-- -----------------------------------------------------

-- -----------------------------------------------------

-- Schema Lessons

-- -----------------------------------------------------

CREATE SCHEMA IF NOT EXISTS `Lessons` DEFAULT CHARACTER SET utf8 ;

USE `Lessons` ;

-- -----------------------------------------------------

-- Table `Lessons`.`teachers`

-- -----------------------------------------------------

CREATE TABLE IF NOT EXISTS `Lessons`.`teachers` (

`idteacher` INT NOT NULL,

`FIO` VARCHAR(100) NULL,

`DATA` DATE NULL,

PRIMARY KEY (`idteacher`))

ENGINE = InnoDB;

-- -----------------------------------------------------

-- Table `Lessons`.`courses`

-- -----------------------------------------------------

CREATE TABLE IF NOT EXISTS `Lessons`.`courses` (

`idcourses` INT NOT NULL,

`title` VARCHAR(100) NULL,

`hour_count` INT NULL,

PRIMARY KEY (`idcourses`))

ENGINE = InnoDB;

-- -----------------------------------------------------

-- Table `Lessons`.`lesson`

-- -----------------------------------------------------

CREATE TABLE IF NOT EXISTS `Lessons`.`lesson` (

`idlesson` INT NOT NULL,

`data` DATE NULL,

`class` CHAR(3) NULL,

`teachers_idteacher` INT NOT NULL,

`courses_idcourses` INT NOT NULL,

PRIMARY KEY (`idlesson`),

INDEX `fk_lesson_teachers_idx` (`teachers_idteacher` ASC),

INDEX `fk_lesson_courses1_idx` (`courses_idcourses` ASC),

CONSTRAINT `fk_lesson_teachers`

FOREIGN KEY (`teachers_idteacher`)

REFERENCES `Lessons`.`teachers` (`idteacher`)

ON DELETE NO ACTION

ON UPDATE NO ACTION,

CONSTRAINT `fk_lesson_courses1`

FOREIGN KEY (`courses_idcourses`)

REFERENCES `Lessons`.`courses` (`idcourses`)

ON DELETE NO ACTION

ON UPDATE NO ACTION)

ENGINE = InnoDB;

SET SQL_MODE=@OLD_SQL_MODE;

SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS;

SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS;

 

Заполним таблицу данными. Запрос на внесение данных:

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('1', 'WEB-мастеринг. Основы серверного программирования.', '100');

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('2', 'Web-маркетинг', '106');

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('3', 'Разработка WEB-приложений ASR.net', '134');

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('4', 'Базы данных', '146');

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('5', 'Основы С++', '96');

INSERT INTO `lessons`.`courses` (`idcourses`, `title`, `hour_count`) VALUES ('6', 'Современные языки программирования', '86');

 

INSERT INTO `lessons`.`teachers` (`idteacher`, `FIO`, `DATA`) VALUES ('1', 'Иванов Иван Иванович', '1982-10-12');

INSERT INTO `lessons`.`teachers` (`idteacher`, `FIO`, `DATA`) VALUES ('2', 'Петров Петр Петрович', '1970-12-12');

INSERT INTO `lessons`.`teachers` (`idteacher`, `FIO`, `DATA`) VALUES ('3', 'Лосев Игорь Викторович', '1975-08-08');

INSERT INTO `lessons`.`teachers` (`idteacher`, `FIO`, `DATA`) VALUES ('4', 'Чуркин Петр Иванович', '1991-01-13');

 

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('1', '2016-11-07', '301', '1', '1');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('2', '2016-11-07', '302', '2', '1');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('3', '2016-11-07', '303', '3', '2');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('4', '2016-11-08', '301', '1', '2');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('5', '2016-11-08', '302', '2', '1');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('6', '2016-11-08', '303', '3', '4');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('7', '2016-11-09', '301', '1', '1');

INSERT INTO `lessons`.`lesson` (`idlesson`, `data`, `class`, `teachers_idteacher`, `courses_idcourses`) VALUES ('8', '2016-11-09', '302', '2', '5');

 

 

Требуется получить:

1. Кто из преподавателей читал курс 30 сентября

SELECT teachers.FIO FROM teachers

INNER JOIN lesson ON teachers.idteacher = lesson.teachers_idteacher

WHERE lesson.data = '2016-11-07';

 

2. Кто из преподавателей читает курсы WEB

SELECT distinct teachers.FIO, courses.title FROM teachers

INNER JOIN lesson ON teachers.idteacher = lesson.teachers_idteacher

INNER JOIN courses ON courses.idcourses = lesson. courses_idcourses

WHERE courses.title LIKE '%web%';

 

3. Какие курсы читает преподаватель Иванов

SELECT distinct courses.title FROM courses

INNER JOIN lesson ON courses.idcourses = lesson. courses_idcourses

INNER JOIN teachers ON teachers.idteacher = lesson.teachers_idteacher

WHERE teachers.FIO = 'Иванов Иван Иванович';

 

4. Когда последний раз преподаватели вели занятия

SELECT teachers.FIO, max(lesson.Data) FROM teachers

INNER JOIN lesson ON teachers.idteacher = lesson.teachers_idteacher

GROUP BY teachers.FIO;

 

5. Если есть преподаватели, которые курсы еще не читали, то записи о них отсутствуют в таблице уроков. Но что делать, если нам они тоже нужны в результате? Тогда соединение в запросе будет не внутренним, а левым:

SELECT teachers.FIO, max(lesson.Data) FROM teachers

LEFT JOIN lesson ON teachers.idteacher = lesson.teachers_idteacher

GROUP BY teachers.FIO;

 








Дата добавления: 2017-08-01; просмотров: 429;


Поиск по сайту:

При помощи поиска вы сможете найти нужную вам информацию.

Поделитесь с друзьями:

Если вам перенёс пользу информационный материал, или помог в учебе – поделитесь этим сайтом с друзьями и знакомыми.
helpiks.org - Хелпикс.Орг - 2014-2024 год. Материал сайта представляется для ознакомительного и учебного использования. | Поддержка
Генерация страницы за: 0.036 сек.