/*+ MATERIALIZE */ 优化器提示在 WITH 子句中的使用验证

Oracle /*+ MATERIALIZE */ 优化器提示在 WITH 子句中的使用验证

概述

/*+ MATERIALIZE */ 是 Oracle 数据库的优化器提示(Hint),核心作用是强制将 WITH 子句(公共表表达式,CTE)的查询结果物化到临时表 中。当后续查询多次引用该 CTE 时,可直接复用临时表数据,避免重复执行子查询;即使仅引用一次,也能通过该 Hint 强制触发物化行为。

测试场景与验证

场景 1:重复引用子查询(非 WITH 子句)------ 无临时表物化

当相同子查询被多次直接引用(未封装到 WITH 子句)时,Oracle 优化器不会将子查询结果物化到临时表,每次引用都会重新执行子查询。

sql 复制代码
SELECT main.cust_id, main.cust_name, main.order_summary, sub1.vip_count
  FROM (
        SELECT c1.cust_id, c1.cust_name, SUM(o.order_amount) AS order_summary
          FROM (SELECT cust_id, cust_name, cust_level
                   FROM customers
                  WHERE total_consume > 50000) c1
          LEFT JOIN orders o
            ON c1.cust_id = o.cust_id
         GROUP BY c1.cust_id, c1.cust_name) main
 CROSS JOIN (
             SELECT COUNT(*) AS vip_count
               FROM (SELECT cust_id, cust_name, cust_level
                        FROM customers
                       WHERE total_consume > 50000) c2
              WHERE c2.cust_level = 'VIP') sub1

执行计划结论 :预估执行计划中未使用临时表空间,子查询被重复执行。

在这里插入图片描述

场景 2:重复引用 WITH 子句中的 CTE------ 触发物化

将重复执行的子查询封装到 WITH 子句中,多次引用该 CTE 时,Oracle 会自动将 CTE 结果物化到临时表。

sql 复制代码
WITH c AS
 (SELECT cust_id, cust_name, cust_level
    FROM customers
   WHERE total_consume > 50000)

SELECT main.cust_id, main.cust_name, main.order_summary, sub1.vip_count
  FROM (
        -- 第一次引用 c
        SELECT c1.cust_id, c1.cust_name, SUM(o.order_amount) AS order_summary
          FROM c c1
          LEFT JOIN orders o
            ON c1.cust_id = o.cust_id
         GROUP BY c1.cust_id, c1.cust_name) main
 CROSS JOIN (
             -- 第二次引用 c
             SELECT COUNT(*) AS vip_count
               FROM c c2
              WHERE c2.cust_level = 'VIP') sub1

执行计划结论 :CTE 的结果集被物化到临时表中,后续引用直接复用临时表数据。

场景 3:单次引用 WITH 子句中的 CTE------ 不触发物化

若 WITH 子句中的 CTE 仅被引用一次,Oracle 优化器默认不会将结果集物化到临时表,而是直接执行子查询。

复制代码
WITH c AS
 (SELECT cust_id, cust_name, cust_level
    FROM customers
   WHERE total_consume > 50000)

SELECT main.cust_id, main.cust_name, main.order_summary
  FROM (
        -- 仅一次引用 c
        SELECT c1.cust_id, c1.cust_name, SUM(o.order_amount) AS order_summary
          FROM c c1
          LEFT JOIN orders o
            ON c1.cust_id = o.cust_id
         GROUP BY c1.cust_id, c1.cust_name) main

执行计划结论:预估执行计划中无临时表物化行为,CTE 子查询直接执行。

场景 4:单次引用 +/*+ MATERIALIZE */ Hint------ 强制物化

在 WITH 子句的 CTE 中添加/*+ MATERIALIZE */ Hint,即使 CTE 仅被引用一次,也能强制 Oracle 将结果集物化到临时表。

测试 SQL

sql

复制代码
WITH c AS
 (SELECT /*+ MATERIALIZE */ cust_id, cust_name, cust_level
    FROM customers
   WHERE total_consume > 50000)

SELECT main.cust_id, main.cust_name, main.order_summary
  FROM (
        -- 仅一次引用 c
        SELECT c1.cust_id, c1.cust_name, SUM(o.order_amount) AS order_summary
          FROM c c1
          LEFT JOIN orders o
            ON c1.cust_id = o.cust_id
         GROUP BY c1.cust_id, c1.cust_name) main

执行计划结论 :CTE 结果集被强制物化到临时表中。

场景 5:Hint 直接写在普通子查询中 ------ 无效

/*+ MATERIALIZE */ Hint 直接添加到非 WITH 子句的普通子查询中,无法触发物化行为。

执行计划结论 :实验验证该方式无效,临时表物化未发生。

三、结论

  1. /*+ MATERIALIZE */ 仅对WITH 子句内的 CTE 生效,直接写在普通子查询中无物化效果;
  2. WITH 子句中的 CTE 被多次引用 时,Oracle 会自动物化结果到临时表;仅被单次引用 时,默认不物化;
  3. 即使 CTE 仅单次引用,也可通过在 WITH 子句的 CTE 查询中添加/*+ MATERIALIZE */ Hint,强制将结果集物化到临时表,适用于优化器判断失误时,未将结果集物化到临时表的情况。
  4. 需要复用子查询结果或优化执行效率的场景。
相关推荐
Apple_羊先森10 小时前
ORACLE数据库巡检SQL脚本--22、检查碎片程度最高的业务表
数据库·sql·oracle
_codemonster13 小时前
PreparedStatement 和 Statement的区别
数据库·oracle
愈努力俞幸运14 小时前
第5章数据库,实体关系图,ER图
数据库·oracle
Apple_羊先森17 小时前
ORACLE数据库巡检SQL脚本--23、检查Oracle数据库中被锁定的数据库对象
数据库·sql·oracle
Apple_羊先森18 小时前
ORACLE数据库巡检SQL脚本--21、正在执行的长耗时操作
数据库·sql·oracle
tryCbest21 小时前
Oracle恢复已损坏定时任务(Jobs)
数据库·oracle
devmoon21 小时前
区块链预言机(Oracle)解析:Polkadot、以太坊与 Solana 如何把现实世界带入链上?
开发语言·oracle·区块链·信息与通信·以太坊·polkadot·solana
德彪稳坐倒骑驴1 天前
数仓中的数据建模方法
数据库·oracle
杨云龙UP1 天前
Oracle ASM归档日志自动清理:RMAN+crontab一键闭环(生产可用)
linux·运维·服务器·数据库·oracle·centos·ux
闲人编程1 天前
聚合管道与复杂查询
开发语言·oracle·lua·match·查询·聚合·lookup