long time lurker, primera pregunta!Mysql slow query: JOIN + multiple WHERES + ORDER POR
estoy luchando para optimizar esta consulta, que selecciona los artículos de precio más bajo que coinciden con los filtros elegidos:
SELECT product_info.*, MIN(product_all.sale_price) as sale_price, product_all.buy_link
FROM product_info
NATURAL JOIN (SELECT * FROM product_all WHERE product_all.date = '2010-09-30') as product_all
WHERE (product_info.category = 2
AND product_info.gender = 'W')
GROUP BY product_all.prod_id
ORDER BY MIN(product_all.sale_price) ASC LIMIT 13
Su explico:
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| 1 | PRIMARY | <derived2> | ALL | NULL | NULL | NULL | NULL | 89801 | Using temporary; Using filesort |
| 1 | PRIMARY | product_info | eq_ref | PRIMARY,category_prod_id_retail_price,category_ret... | PRIMARY | 4 | product_all.prod_id | 1 | Using where |
| 2 | DERIVED | product_all | ref | date_2 | date_2 | 3 | | 144107 | |
He intentado eliminar la sub consulta, la cual intuitivamente parece mejor, pero en la práctica lleva aún más tiempo:
SELECT product_info.*, MIN(product_all.sale_price) as sale_price, product_all.buy_link
FROM product_info
NATURAL JOIN product_all
WHERE (product_all.date = '2010-09-30'
AND product_info.category = 2
AND product_info.gender = 'W')
GROUP BY product_all.prod_id
ORDER BY MIN(product_all.sale_price) ASC LIMIT 13
Y su exp yacido:
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| 1 | SIMPLE | product_info | ref | PRIMARY,category_prod_id_retail_price,category_ret... | category_retail_price | 5 | const | 269 | Using where; Using temporary; Using filesort |
| 1 | SIMPLE | product_all | ref | PRIMARY,prod_id,date_2 | prod_id | 4 | equipster_db.product_info.prod_id | 141 | Using where |
Estas son las tablas:
CREATE TABLE `product_all` (
`prod_id` INT(10) NOT NULL PRIMARY KEY ,
`ref_id` INT(10) NOT NULL PRIMARY KEY ,
`date` DATE NOT NULL ,
`buy_link` BLOB NOT NULL ,
`sale_price` FLOAT NOT NULL
) ENGINE = MYISAM ;
CREATE TABLE `product_info` (
`prod_id` INT(10) NOT NULL AUTO_INCREMENT PRIMARY KEY ,
`prod_name` VARCHAR(200) NOT NULL,
`brand` VARCHAR(50) NOT NULL,
`retail_price` FLOAT NOT NULL
`category` INT(3) NOT NULL,
`gender` VARCHAR(1) NOT NULL,
`type` VARCHAR(10) NOT NULL
) ENGINE = MYISAM ;
Mis preguntas: estructura
-que consulta parece óptima?
¿Qué índices optimizarían esta consulta?
-menos importante: ¿cómo cambia el enfoque de indexación al añadir o eliminar cláusulas WHERE o el uso de un orden diferente BY, tales como la clasificación por% de descuento:
ORDER BY (1-(MIN(product_all.sale_price)/product_info.retail_price)) DESC
edición: ambas consultas de reunión natural actúa sobre prod_id (un registro en product_info puede tener varias instancias en product_all, por lo que deben agruparse)
uno de los PK's es compuesto, pero sí cada grupo es una fila: el ID del producto, el precio más bajo para ese producto y datos relacionados. editar: esta fue una respuesta a un comentario que parece haberse desvanecido. edit2: sí, creo que presioné editar en lugar de agregar comentario ... sin problemas. – chrisblanch