似たような質問がたくさんあることは知っていますが、この質問の実際の解決策を見つけるのに 2 週間かかりました。
このクエリは、POS プログラムからのものです。このクエリは、ユーザーが必要なレポート (期間、合計など) を選択し、結果をグループ化するかどうかを選択するフォームに関連しています。
これは不便です。クエリは VB.NET で、つまりコードで生成され、ユーザーの選択 (さまざまな合計、期間、グループなど) によって異なります。 " フォームの他のすべてのクエリをコードで指定します。
この場合、このクエリはファミリ別にグループ化された合計クエリです。
ほとんどの時間 (>99%) はデータの送信に費やされています (クエリ # のプロファイルを表示)
表は次のとおりです。
CREATE TABLE `product` (
`idProduct` smallint(5) unsigned NOT NULL AUTO_INCREMENT,
`idFamily` tinyint(3) unsigned DEFAULT NULL,
`Codigo` char(10) NOT NULL,
`Nombre` char(70) DEFAULT NULL COMMENT 'Nombre corto',
-- five more integer columns
PRIMARY KEY (`idProduct`),
KEY `fk_p_idFamily` (`idFamily`),
CONSTRAINT `fk_p_idFamily` FOREIGN KEY (`idFamily`) REFERENCES `family` (`idFamily`),
) ENGINE=InnoDB AUTO_INCREMENT=19420 DEFAULT CHARSET=latin1 PACK_KEYS=0;
CREATE TABLE `family` (
`idFamily` tinyint(3) unsigned NOT NULL AUTO_INCREMENT,
`Nombre` char(30) NOT NULL,
`Descripcion` char(255) DEFAULT NULL,
`Borrado` tinyint(1) NOT NULL DEFAULT '0',
PRIMARY KEY (`idFamily`)
) ENGINE=InnoDB AUTO_INCREMENT=34 DEFAULT CHARSET=latin1 PACK_KEYS=0
CREATE TABLE `document` (
`idDocument` tinyint(3) unsigned NOT NULL AUTO_INCREMENT,
`Nombre` char(25) NOT NULL,
`Descripcion` char(100) DEFAULT NULL,
`Borrado` tinyint(1) NOT NULL DEFAULT '0',
`NoComputa` tinyint(1) NOT NULL DEFAULT '0',
`Rectifica` tinyint(1) NOT NULL DEFAULT '0',
`CalculoSumatorioPVP` tinyint(1) NOT NULL DEFAULT '0',
PRIMARY KEY (`idDocument`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=latin1 PACK_KEYS=0
CREATE TABLE `soldproduct` (
`idProduct` smallint(5) unsigned NOT NULL,
`idSale` int(10) unsigned NOT NULL,
`PrecioCompra` decimal(7,2) NOT NULL ,
`PrecioVenta` decimal(7,2) NOT NULL ,
`DtoProd` decimal(7,4) DEFAULT NULL ,
`BrutoUd` decimal(7,2) NOT NULL ,
`PVPUd` decimal(7,2) NOT NULL ,
`Cantidad` decimal(9,3) DEFAULT NULL ,
PRIMARY KEY (`idProduct`,`idSale`),
KEY `fk_pv_idSale` (`idSale`),
CONSTRAINT `fk_pv_idProduct` FOREIGN KEY (`idProduct`) REFERENCES `product` (`idProduct`),
CONSTRAINT `fk_pv_idSale` FOREIGN KEY (`idSale`) REFERENCES `sales` (`idSale`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1 PACK_KEYS=0
CREATE TABLE `sales` (
`idSale` int(10) unsigned NOT NULL AUTO_INCREMENT,
`idDocument` tinyint(3) unsigned NOT NULL,
`idEstadoVenta` tinyint(3) unsigned NOT NULL,
`idCliente` smallint(5) unsigned NOT NULL,
`idFormaPago` tinyint(3) unsigned NOT NULL,
`idEmpleado` tinyint(3) unsigned NOT NULL ,
`idTienda` tinyint(3) unsigned DEFAULT NULL,
`idTipoVenta` tinyint(3) unsigned NOT NULL,
`FechaVenta` datetime DEFAULT NULL COMMENT 'Fecha de Venta',
`PrecioCompraTotal` decimal(10,2) DEFAULT NULL,
`IVA` decimal(7,4) DEFAULT NULL,
-- ten more decimal columns
-- five more datetime columns
-- ten more char columns
`Borrado` tinyint(1) NOT NULL DEFAULT '0' ,
`Historia` tinyint(1) NOT NULL DEFAULT '0',
PRIMARY KEY (`idVenta`),
KEY `fk_v_idTienda` (`idTienda`),
KEY `fk_v_idCliente` (`idCliente`),
KEY `fk_v_idEmpleado` (`idEmpleado`),
KEY `fk_v_idTipoVenta` (`idTipoVenta`),
KEY `fk_v_idFormaPago` (`idFormaPago`),
KEY `fk_v_idDocument` (`idDocument`),
KEY `fk_v_idEstadoVenta` (`idEstadoVenta`),
KEY `idx_v_FechaVenta` (`FechaVenta`),
CONSTRAINT `fk_v_idCliente` FOREIGN KEY (`idCliente`) REFERENCES `cliente` (`idCliente`),
CONSTRAINT `fk_v_idDocument` FOREIGN KEY (`idDocument`) REFERENCES `document` (`idDocument`),
CONSTRAINT `fk_v_idEmpleado` FOREIGN KEY (`idEmpleado`) REFERENCES `empleado` (`idEmpleado`),
CONSTRAINT `fk_v_idEstadoVenta` FOREIGN KEY (`idEstadoVenta`) REFERENCES `estadoventa` (`idEstadoVenta`),
CONSTRAINT `fk_v_idFormaPago` FOREIGN KEY (`idFormaPago`) REFERENCES `formapago` (`idFormaPago`),
CONSTRAINT `fk_v_idTienda` FOREIGN KEY (`idTienda`) REFERENCES `tienda` (`idTienda`),
CONSTRAINT `fk_v_idTipoVenta` FOREIGN KEY (`idTipoVenta`) REFERENCES `tipoventa` (`idTipoVenta`)
) ENGINE=InnoDB AUTO_INCREMENT=101770 DEFAULT CHARSET=latin1 PACK_KEYS=0
クエリは次のとおりです。
SELECT f.Nombre ,SUM(sp.PrecioVenta*sp.Cantidad)
FROM soldproduct sp, sales s, document doc, family f, product p
WHERE s.idDocument = doc.idDocument AND doc.NoComputa = FALSE
AND p.idProduct = sp.idProduct AND sp.idSale = s.idSale
AND p.idFamily = f.idFamily AND p.Borrado = FALSE
AND s.Borrado = FALSE AND s.Historia = FALSE AND s.idTienda = 1
AND s.FechaVenta BETWEEN '2013-01-01' AND '2014-01-01' GROUP BY f.idFamily;
私もこれを試しました(責任がある場合に備えて、ドキュメントテーブルも削除しました)
SELECT ProductFamily.Nombre, SUM(sp.PrecioVenta*sp.Cantidad)
FROM
(SELECT idSale FROM sales WHERE Borrado = FALSE AND Historia = FALSE AND idTienda = 1
AND FechaVenta BETWEEN '2013-01-01' AND '2014-01-01') SalesidSale
JOIN
soldproduct sp
ON sp.idSale = SalesidSale.idSale
JOIN
(SELECT p.idProduct, p.idFamily, f.Nombre FROM product p, family f WHERE
p.idFamily = f.idFamily AND p.Borrado = FALSE) ProductFamily
ON ProductFamily.idProduct = sp.idProduct
GROUP BY ProductFamily.idFamily;
かかる時間は非常に長く、EXPLAIN コマンドへの出力は (最初のクエリ):
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE v range PRIMARY,fk_v_idTienda,fk_v_idDocument,idx_v_FechaVenta idx_v_FechaVenta 6 NULL 7387 "Using index condition; Using where; Using MRR; Using temporary; Using filesort"
1 SIMPLE doc ALL PRIMARY NULL NULL NULL 4 "Using where; Using join buffer (Block Nested Loop)"
1 SIMPLE pv ref PRIMARY,fk_pv_idSale fk_pv_idSale 4 gemalia.s.idSale 4 NULL
1 SIMPLE p eq_ref PRIMARY,fk_p_idFamily PRIMARY 2 gemalia.sp.idProduct 1 "Using where"
1 SIMPLE f eq_ref PRIMARY PRIMARY 1 gemalia.p.idFamily 1 NULL
誰かが私を助けてくれることを願っています.インデックス、サブクエリなどを作成しようとしました..しかし、40秒未満のものは取得できません..
テーブルあたりの行数は、およそ次のとおりです。 売上: 100,000 製品: 20,000 家族: 35 販売された製品: 1,100,000 ドキュメント: 4
どうもありがとう。