Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
для студентов РУБД / теория по SQL 1 часть.doc
Скачиваний:
47
Добавлен:
21.03.2016
Размер:
631.3 Кб
Скачать

Максимальное значение столбца

SELECT MAX(article) AS article FROM shop

article

4

"Как определить номер, дилера и цену самого дорогого изделия?"

SELECT article, dealer, price FROM shop WHERE price=(SELECT MAX(price) FROM shop)

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

SELECT article, dealer, price FROM shop WHERE price=19.95

Существует еще одно решение: отсортировать все строки по убыванию цен и после этого получить первую строку, используя специальный оператор LIMIT:

SELECT article, dealer, priceFROM shop ORDER BY price DESC LIMIT 1

Примечание: если существует несколько самых дорогих изделий (например, каждое из них стоит 19,95), запрос, использующий LIMIT, возвращает лишь одно из них!

Как определить наибольшую цену по каждому изделию?

SELECT article, MAX(price) AS price FROM shop GROUP BY article

article

|price

0001

3.99

0002

10.99

0003

1.69

0004

19.95

Для каждого изделия, как определить дилер(ов) с самыми высокими ценами?

SELECT article, dealer, price FROM shop s1 WHERE price=

(SELECT MAX(s2.price) FROM shop s2 WHERE s1.article = s2.article);

1.Следует получить список (изделие, максимальная цена)

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

Это легко делается с помощью временной таблицы:

CREATE TEMPORARY TABLE tmp ( article INT(4) UNSIGNED ZEROFILL DEFAULT '0000' NOT NULL, price DOUBLE(16,2) DEFAULT '0.00' NOT NULL);

LOCK TABLES shop read;

INSERT INTO tmp SELECT article, MAX(price) FROM shop GROUP BY article;

SELECT shop.article, dealer, shop.price FROM shop, tmp

WHERE shop.article=tmp.article AND shop.price=tmp.price;

UNLOCK TABLES;

DROP TABLE tmp;

Если не использовать ключевое слово TEMPORARY, следует поставить блокировку на таблицу tmp.

"А можно ли это сделать одним запросом?"

SELECT article, SUBSTRING( MAX( CONCAT(LPAD(price,6,'0'),dealer) ), 7) AS dealer,00+LEFT( MAX( CONCAT(LPAD(price,6,'0'),dealer) ), 6) AS price FROM shop GROUP BY article;

article

dealer

price

0001

B

3.99

0002

A

10.99

0003

C

1.69

0004

D

19.95

Использование пользовательских переменных

В MySQL для хранения результатов, чтобы не держать их во временных переменных на клиенте, можно применять пользовательские переменные

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

mysql> SELECT @min_price:=MIN(price),@max_price:=MAX(price) FROM shop;

mysql> SELECT * FROM shop WHERE price=@min_price OR price=@max_price;

article

dealer

price

0003

D

1.25

0004

D

19.95

Поиск по двум ключам

MySQL пока не осуществляет оптимизации, если поиск производится по двум различным ключам, которые связаны при помощи оператора OR (поиск по одному ключу с различными частями OR оптимизируется хорошо):

SELECT field1_index, field2_index FROM test_table WHERE field1_index = '1'OR field2_index = '1'

В настоящее время данную проблему очень эффективно можно решить при помощи временной таблицы. Этот способ оптимизации также хорошо подходит, если вы запускаете очень сложные запросы, когда SQL-сервер делает оптимизацию в неправильном порядке.

CREATE TEMPORARY TABLE tmp

SELECT field1_index, field2_index FROM test_table WHERE field1_index = '1';

INSERT INTO tmp

SELECT field1_index, field2_index FROM test_table WHERE field2_index = '1';

SELECT * from tmp;

DROP TABLE tmp;

Вышеупомянутый способ выполнения этого запроса - это фактически UNION (объединение) двух запросов.

Подсчет посещений за день

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

CREATE TABLE t1 (year YEAR(4), month INT(2) UNSIGNED ZEROFILL,day INT(2) UNSIGNED ZEROFILL);

INSERT INTO t1 VALUES(2000,1,1),(2000,1,20),(2000,1,30),(2000,2,2),(2000,2,23),(2000,2,23);

SELECT year,month,BIT_COUNT(BIT_OR(1<<day)) AS days FROM t1 GROUP BY year,month;

year

month

days

2000

01

3

2000

02

2

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