SQL Server的优化器会缓存标量子查询结果集吗

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了SQL Server的优化器会缓存标量子查询结果集吗相关的知识,希望对你有一定的参考价值。

在这篇博客ORACLE当中自定义函数性优化浅析中,我们介绍了通过标量子查询缓存来优化函数性能: 标量子查询缓存(scalar subquery caching)会通过缓存结果减少SQL对函数(Function)的调用次数, ORACLE会在内存中构建一个哈希表来缓存标量子查询的结果。 那么SQL Server的优化器是否也会有类似这样的功能呢? 抱着这样的疑问,动手测试了一下,准备测试环境

 

 

CREATE TABLE TEST
(
   ID  INT
);
 
 
DECLARE @RowIndex INT =1;
 
WHILE @RowIndex <= 8 
BEGIN
    INSERT INTO TEST
    SELECT @RowIndex ;
    
    SET  @RowIndex = @RowIndex +1;
END

 

 

然后创建函数SLOW_FUNCTION, 本想在函数里面使用WAITFOR DELAY延迟2秒构造那种性能开销较大的函数,来模拟达到实验效果。但是标量函数里面不允许使用WAITFOR DELAY,报Invalid use of a side-effecting operator \'WAITFOR\' within a function.

 

 

CREATE  FUNCTION SLOW_FUNCTION(@p_value INT )
RETURNS INT
AS
BEGIN
    WAITFOR DELAY \'00:00:00.002\';
    RETURN @p_value+10;
END;

 

 

那么我就变相构造一个这样的函数,用一个循环一直延迟2秒后,标量函数才返回执行结果。

 

 

 
DROP FUNCTION SLOW_FUNCTION;
GO
CREATE  FUNCTION SLOW_FUNCTION ( @p_value INT )
RETURNS INT
AS
    BEGIN
        DECLARE @dt_start DATETIME;
        DECLARE @dt_end DATETIME;
 
        SET @dt_start = GETDATE();
        SET @dt_end = DATEADD(ss, 2, GETDATE())
        WHILE @dt_start < @dt_end
            SET @dt_start = GETDATE();
 
        RETURN @p_value+10;
    END;

 

 

 

 

clip_image001

 

 

构造出现重复数据的情况,然后测试对比,测试对比发现,在SQL Server中,优化器根本不会缓存子查询结果集。这种优化函数的技术在SQL Server中根本行不通。优化器根本没有这样的优化功能。

 

 

 

TRUNCATE TABLE TEST;

GO

INSERT INTO TEST

SELECT 1  UNION ALL

SELECT 1  UNION ALL

SELECT 1  UNION ALL

SELECT 2  UNION ALL

SELECT 2  UNION ALL

SELECT 2  UNION ALL

SELECT 3  UNION ALL

SELECT 3;

 

clip_image002

以上是关于SQL Server的优化器会缓存标量子查询结果集吗的主要内容,如果未能解决你的问题,请参考以下文章

SQL 查询优化:标量子查询是不是会影响性能?

T-SQL,在视图中重复相同的标量子查询性能

子查询优化

SQL为王:oracle标量子查询和表连接改写

SQL优化:慎用标量子查询,改用left join提升查询效率

MySQL随记 - 子查询