gpt4 book ai didi

Mysql 查询优化,将 CASE Exists (SELECT...) 移动到左连接中

转载 作者:行者123 更新时间:2023-11-29 02:35:38 24 4
gpt4 key购买 nike

更新这里是CASE语句使用的两个表的ddl

CREATE TABLE product_option (
merchant_id smallint unsigned NOT NULL,
product_option_id smallint unsigned NOT NULL AUTO_INCREMENT,
product_option_name varchar(255) NOT NULL DEFAULT '',
product_id int unsigned NOT NULL DEFAULT 0, /* Option may be associated with a product */
package_id int unsigned NOT NULL DEFAULT 0, /* or all products with this package */
PRIMARY KEY pk_product_option (merchant_id,product_option_id),

) TYPE=MyISAM PACK_KEYS=1 COMMENT='Options';

CREATE TABLE package_2_product (
merchant_id smallint unsigned NOT NULL,
package_id int unsigned NOT NULL,
product_id int unsigned NOT NULL,

PRIMARY KEY pk_package_2_product (merchant_id,package_id,product_id)

) TYPE=MyISAM PACK_KEYS=1 COMMENT='Link product to package';

我有一个问题...

SELECT SQL_CALC_FOUND_ROWS 
p.*,
CASE p.in_stock_msg WHEN '' THEN 'In stock' ELSE p.in_stock_msg END AS in_stock_msg,
CASE p.out_stock_msg WHEN '' THEN '' ELSE p.out_stock_msg END AS out_stock_msg,
CASE WHEN EXISTS (
SELECT product_option_id
FROM product_option
WHERE merchant_id = 116 AND product_id = p.product_id
UNION
SELECT product_id FROM package_2_product
WHERE merchant_id = 116 AND product_id = p.product_id
) THEN 1 ELSE 0 END AS options_exist,
i.thumbnail,i.thumbnail_width,i.thumbnail_height,
i.title AS thumbnail_title, i.alt AS thumbnail_alt
FROM
product p
INNER JOIN category_2_product c2p ON p.merchant_id=c2p.merchant_id
AND p.product_id=c2p.product_id
AND c2p.category_id = 84
LEFT JOIN product_image i ON p.merchant_id = i.merchant_id
AND p.product_id = i.product_id
AND i.is_default = 1
WHERE
p.merchant_id = 116
AND FIND_IN_SET('live',p.param) > 0
AND FIND_IN_SET('wholesale-only',p.param) = 0
ORDER BY
p.rank, p.product_name
LIMIT 0, 50;

相关部分是存在时的情况...导致查询需要多秒才能运行。如果没有子选择,它会在十分之一秒内完成。

我想知道是否有一种方法可以移动这种类型的案例,子选择进入内部联接以获得相同的数据,而不是对返回的每一行都运行子选择。

通过调整 Coalesce 行和 LEFT JOIN (SELECT...) p1 尝试以下查询

我收到这个错误...

错误代码:1054“on 子句”中的未知列“p1.product_id”

这里是查询...

SELECT SQL_CALC_FOUND_ROWS 
p.*,
CASE p.in_stock_msg WHEN '' THEN 'In stock' ELSE p.in_stock_msg END AS in_stock_msg,
CASE p.out_stock_msg WHEN '' THEN '' ELSE p.out_stock_msg END AS out_stock_msg,
COALESCE(p1.product_option_id, p2.product_id) IS NOT NULL AS options_exist,
i.thumbnail,i.thumbnail_width,i.thumbnail_height,
i.title AS thumbnail_title, i.alt AS thumbnail_alt
FROM
product p
INNER JOIN category_2_product c2p ON p.merchant_id=c2p.merchant_id
AND p.product_id=c2p.product_id
AND c2p.category_id = 84
LEFT JOIN product_image i ON p.merchant_id = i.merchant_id
AND p.product_id = i.product_id
AND i.is_default = 1
LEFT JOIN (
SELECT product_option_id
FROM product_option
WHERE merchant_id = 116
GROUP BY product_id) p1 on p1.product_id = p.product_id
LEFT JOIN (
SELECT product_id
FROM package_2_product
WHERE merchant_id = 116
GROUP BY product_id) p2 on p1.product_id is null and p2.product_id = p.product_id
WHERE
p.merchant_id = 116
AND FIND_IN_SET('live',p.param) > 0
AND FIND_IN_SET('wholesale-only',p.param) = 0
ORDER BY
p.rank, p.product_name

最佳答案

SELECT SQL_CALC_FOUND_ROWS 
p.*,
CASE p.in_stock_msg WHEN '' THEN 'In stock' ELSE p.in_stock_msg END AS in_stock_msg,
CASE p.out_stock_msg WHEN '' THEN '' ELSE p.out_stock_msg END AS out_stock_msg,
COALESCE(p1.product_id, p2.product_id) IS NOT NULL AS options_exist,
i.thumbnail,i.thumbnail_width,i.thumbnail_height,
i.title AS thumbnail_title, i.alt AS thumbnail_alt
FROM
product p
INNER JOIN category_2_product c2p ON p.merchant_id=c2p.merchant_id
AND p.product_id=c2p.product_id
AND c2p.category_id = 84
LEFT JOIN product_image i ON p.merchant_id = i.merchant_id
AND p.product_id = i.product_id
AND i.is_default = 1
LEFT JOIN (
SELECT product_id
FROM product_option
WHERE merchant_id = 116
GROUP BY product_id) p1 on p1.product_id = p.product_id
LEFT JOIN (
SELECT product_id
FROM package_2_product
WHERE merchant_id = 116
GROUP BY product_id) p2 on p1.product_id is null and p2.product_id = p.product_id

WHERE
p.merchant_id = 116
AND FIND_IN_SET('live',p.param) > 0
AND FIND_IN_SET('wholesale-only',p.param) = 0
ORDER BY
p.rank, p.product_name

注意事项:

  • on p1.product_id is null 添加到第二个左连接以防止它在 p1 已经有结果时被执行。
  • group by 子句防止 LEFT JOIN 通过笛卡尔积扩展结果集


此 DDL 将创建一个包含足够字段的表,以显示查询正常工作。

create table product_option(merchant_id int, product_id int);
create table package_2_product(merchant_id int, product_id int);
create table category_2_product(merchant_id int, product_id int, category_id int);
create table product_image(merchant_id int, product_id int, is_default int,
thumbnail int, thumbnail_width int, thumbnail_height int, title int, alt int);
create table product(merchant_id int, product_id int, in_stock_msg int,
out_stock_msg int, param int, rank int, product_name int);

关于Mysql 查询优化,将 CASE Exists (SELECT...) 移动到左连接中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5490350/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com