拿到一个sql,同事告诉我这个sql索引加了,为什么还是这么慢。
sale_order_goods,sale_order 两张表都有几十万的数据。其中in结果集有30万。
sql如下:
SELECT
ifnull(sum(buy_number), 0) AS buy_number_sum
FROM
sale_order_goods
WHERE
sale_order_id IN (
SELECT
so.id
FROM
sale_order_goods sog,
sale_order so
WHERE
1 = 1
AND sog.tenant_org_id =1
AND so.tenant_org_id =1
AND sog.sale_order_id = so.id
AND so.order_status =1
AND DATE_SUB(CURDATE(), INTERVAL 30 DAY) <= date(sog.created_at)
);
这是一个in + 子查询的语句。
顺便提一句,innodb对in + 子查询的处理非常差,比如:
select * from a where a.oid in (select id from b where b.name='xxx')
,如果
select id from b where b.name='xxx'
返回的结果是1,2,3。那么sql是不是就变成了
select * from a where a.oid in (1,2,3)
. NO!!,事实上不是这样。innodb会把外层表压入到子查询中,sql会变成
select * from a exists (select * from b where name='xxx' and b.id = a.oid)
这时,子查询关联外部表a,所以innodb认为无法执行这个子查询,于是需要全表遍历一遍a,再根据a返回的oid逐个执行一次子查询。
上面这个sql不需要执行子查询,改写如下:
SELECT
ifnull(sum(buy_number), 0) AS buy_number_sum
FROM
sale_order_goods sog,
sale_order so
WHERE
1 = 1
AND sog.tenant_org_id =1
AND so.tenant_org_id =1
AND sog.sale_order_id = so.id
AND so.order_status =1
AND DATE_SUB(CURDATE(), INTERVAL 30 DAY) <= date(sog.created_at)
直接放弃 in + 子查询的方式。sql在1秒内执行完毕。
分享到:
相关推荐
理解子查询的使用和优化是编写复杂查询的必备技能。 “数据聚合和透视”涉及到GROUP BY、HAVING和聚合函数(如COUNT、SUM、AVG、MIN和MAX)。这些功能用于汇总数据,而透视则允许我们从不同角度查看数据,例如通过...
15. **使用EXISTS而非IN**:当子查询仅用于判断是否存在匹配记录时,EXISTS通常比IN更快。 16. **使用UNION而非UNION ALL**:只有在确保没有重复记录时才使用UNION ALL,因为它不进行去重操作,速度更快。 17. **...
- **IN**: 检查值是否存在于子查询结果中。 - **EXISTS**: 检查子查询是否有结果。 ##### 2.10 外联结、交叉联结和自联结 - **LEFT OUTER JOIN**: 包含左表的所有记录。 - **CROSS JOIN**: 两个表的所有组合。 - **...
#### 一、Oracle SQL优化概览 在系统开发与维护过程中,随着数据库数据量的增长,SQL语句的性能优化成为提升系统响应速度的关键环节。优化SQL不仅能够显著提升数据处理效率,还能大幅节省硬件资源,减少运维成本。...
- **子查询优化**:子查询效率取决于结果集大小,与连接查询相比,各有优势,具体场景需灵活选择。 5. **数据应用**: - **DDL语句**:包括创建和删除数据库、创建和修改表结构等操作。 - **DML语句**:涉及数据...
SQL优化能够显著提升系统的响应速度,尤其是在数据库中数据量剧增的情况下,良好的SQL编写技巧可以使得查询速度提升上百倍。 首先,要注意SQL语句的编写规范。在多表联接查询时,应考虑表的连接顺序。通常,应将...
2. **子查询**:在查询中嵌套另一个查询,通常用于提供临时结果集。例如: ```sql SELECT column1 FROM table1 WHERE column1 IN (SELECT column2 FROM table2); ``` 3. **联接(UNION)操作**:用于合并两个或...
子查询可以用于比较、联接、集合操作等,例如在WHERE子句中使用IN、NOT IN、ANY、ALL等操作符。 2. **连接查询(JOIN)**:连接查询允许合并来自两个或更多表的数据。Oracle支持多种类型的JOIN,包括INNER JOIN、...
在子查询中,应当在子查询前后加上括号,例如 `SELECT col1, col2 FROM table_a WHERE col3 IN (SELECT col4 FROM table_b WHERE col4 > 0)`。 **10. 避免使用无意义的表达式** 避免在WHERE子句中使用诸如`1=1`或`...
- **使用EXISTS代替IN**:当查询涉及子查询时,使用EXISTS通常比IN更高效,因为前者可以在找到第一个匹配项后立即停止搜索。 - **避免SELECT * FROM**:指定所需的列名而非使用星号(*),这样可以减少不必要的...
在进行SQL优化的过程中,Oracle中的`HINT`是一种非常实用且强大的工具,它允许数据库管理员或开发者通过在SQL语句中添加注释的方式直接指导Oracle的优化器如何执行查询计划。这种方式可以有效地调整数据访问路径、...
然而,通过巧妙运用`CASE WHEN`语句,可以将这些子查询合并为一个更加简洁高效的查询语句,从而减少表扫描次数,提升查询性能。 原始SQL: ```sql SELECT DISTINCT ne_state.peer_id peer_name, to_char(ne_state....
1. **NOT IN子查询**: ```sql SELECT * FROM a_matrix_navigation_map WHERE rowid NOT IN (SELECT rowid FROM a_matrix_navigation_map WHERE rownum ) AND rownum ; ``` 2. **嵌套查询**: ```sql ...
- SQL查询语句`SELECT id, je FROM sy WHERE id IN (SELECT id FROM zc WHERE bbq='200704' AND ts>rc) AND bbq='200704' ORDER BY je DESC`是基于两个表(Sy和Zc)的数据筛选,使用了子查询(IN子句)和ORDER BY...
本文介绍了在SqlServer2008环境下进行查询优化的方法和技巧,包括Sargability规则、AND/OR操作符的调优、用NOT EXISTS代替NOT IN以及避免使用COUNT(*)的子查询等。通过对这些规则的实际应用和实验验证,证明了它们...
- **子查询**:嵌套在另一个SQL查询中的查询。 - 子查询可以用于SELECT、INSERT、UPDATE和DELETE语句中。 - 根据子查询返回的结果数量,可以分为标量子查询、行子查询和表子查询。 #### 9. 特定数据类型查询 - ...
- **HASH_AJ(v)** 和 **MERGE_AJ(v)**: 控制使用散列反连接或合并反连接处理NOT IN子查询。 - **HASH_SJ(v)** 和 **MERGE_SJ(v)**: 控制使用散列或合并半连接处理EXISTS子查询。 - **PUSH_JOIN_PRED(v)** 和 **NO_...
#### 一、SQL优化概述 在数据库管理系统中,SQL(Structured Query Language)是用于管理和处理数据的核心语言。随着数据量的增长和技术的发展,如何提高SQL执行效率成为了一个重要的课题。本章节将从多个方面详细...
尤其是在子查询中使用它们时更应该小心。可以考虑用 `EXISTS` 或 `NOT EXISTS` 替代 `IN` 和 `NOT IN`,以提高查询性能。 - **使用 `IS NULL` 和 `IS NOT NULL`**:相比于 `column IS NULL` 和 `column IS NOT NULL`...