子查询在存储过程中不自动缓存,每次执行均重算;应优先使用临时表显式缓存中间结果,并建索引优化后续访问,CTE仅是语法糖不保证物化。子查询在存储过程中不自动缓存,重复执行等于重复查表SQL Server、MySQL 或 PostgreSQL 的存储过程里,写两次一样的 SELECT ... FROM (subquery) AS t,数据库不会帮你"记住"第一次的结果。每次遇到就重跑一遍------哪怕子查询耗时、扫描上百万行、还带聚合或 JOIN。这不是 bug,是标准行为:SQL 引擎按需求值,没有隐式中间结果生命周期管理。实操建议:用 WITH(CTE)把子查询提成命名临时结果,但注意:CTE 在多数引擎中只是语法糖,不保证物化(PostgreSQL 12+ 加 MATERIALIZED 才强制;SQL Server 需配合 OPTION (RECOMPILE) 和统计信息才可能复用)真要缓存,得显式落地:用 CREATE TABLE #temp(SQL Server)或 CREATE TEMPORARY TABLE(MySQL/PG),再 INSERT INTO ... SELECT 一次,后续全查这个临时表别在循环体里反复调用含子查询的视图或表值函数------每次调用都重算,不是"查缓存"临时表比 CTE 更可靠地复用中间结果当你需要多次引用同一组计算结果(比如用户标签聚合、订单状态快照),WITH 看起来干净,但执行计划里常被展开成多次子查询。而临时表是物理存在的,插入后数据就在内存或 tempdb 里,后续 SELECT 直接走索引或顺序扫描,开销可控。实操建议:SQL Server:优先用 #temp_table,建好后立刻加索引(CREATE INDEX IX ON #temp(col)),尤其对后续 JOIN 或 WHERE 过滤的字段MySQL:用 CREATE TEMPORARY TABLE tmp AS SELECT ...,注意它不支持外键和全文索引,但能用 INDEX 语句显式建索引PostgreSQL:用 CREATE TEMP TABLE tmp AS SELECT ...,默认不记录 WAL,速度快;如需排序/JOIN 性能,手动 CREATE INDEX ON tmp(col)别省那句 DROP TABLE #temp(SQL Server)或依赖会话结束自动清理------长事务或连接池复用下,残留临时表可能引发冲突WHERE 条件下推进子查询反而破坏缓存机会有人为了"提前过滤"把外层 WHERE 拆进子查询里,比如把 SELECT * FROM (SELECT u.id, u.name FROM users u) t WHERE t.id IN (1,2,3) 改成 SELECT * FROM (SELECT u.id, u.name FROM users u WHERE u.id IN (1,2,3)) t。这看似减少数据量,但会让子查询失去通用性------一旦外层条件变,就得重写整个子查询,没法复用临时表或预计算结果。 RedClaw 百度推出的手机端万能AI Agent助手
相关推荐
薪火铺子11 分钟前
MySQL InnoDB 索引底层:B+树深度解析Soari12 分钟前
深度办公革命:拆解 Claude for Microsoft 365,打造金融级智能办公生态Elastic 中国社区官方博客16 分钟前
从平均值到任意百分位数:Elasticsearch 在 ES|QL 中原生支持指数直方图.唉22 分钟前
06. FastAPI框架从入门到实战276695829235 分钟前
某白山小程序限制PC端调试今儿敲了吗38 分钟前
数据库(四)——关系数据库SQL语言XerCis39 分钟前
ngrok实现内网穿透(以Python FastAPI为例)Json____41 分钟前
Python练习题集-实用小工具与数据应用brevity_souls42 分钟前
SQL server格式化日期虹科网络安全43 分钟前
艾体宝干货|Active-Active/Active-Passive 数据库架构解析:高可用设计中的权衡与选型