子查询在存储过程中不自动缓存,每次执行均重算;应优先使用临时表显式缓存中间结果,并建索引优化后续访问,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助手
相关推荐
coderwei1232 小时前
从OpenAI到Strip:用六大支柱读懂Harness Engineering的生产实践海鸥-w2 小时前
Python(FastAPI)中ORM框架Sqlalchemy的安装及建表爱喝水的鱼丶2 小时前
SAP-ABAP:SAP基础数据校验工具开发系列博客(共5篇)第三篇:SAP接口对接开发:实现数据的实时/批量校验交互真香号3 小时前
记一次生产RocketMQ消息积压消费慢的排查与解决数据库小学妹3 小时前
国产数据库技术成熟度实测:从Oracle兼容到高可用,四个维度评估能不能上生产Wonderful U3 小时前
Python+Django实战|个人博客内容管理系统:搭建轻量化、高自由度的个人动态博客CMS系统JdSnE27zv3 小时前
数据库性能优化三:程序操作优化高洁013 小时前
智能体:你的私人数字助理海鸥-w3 小时前
python(fastapi) 实现更新,新增,删除接口淘矿人3 小时前
DeepSeek V4对决Claude 4.8:AI模型终极横评