3

我有一个表值的内联 UDF。我想过滤该 UDF 的结果以获得一个特定的值。当我使用常量参数指定过滤器时,一切都很好,性能几乎是瞬时的。当我使用可变参数指定过滤器时,它需要大量的时间,逻辑读取量增加 500 倍,持续时间增加 20 倍。

执行计划表明,在可变参数情况下,直到过程的很晚才应用过滤器,导致多次索引扫描,而不是在常量情况下执行的查找。

我想我的问题是:为什么,因为我指定了一个对索引字段具有高度选择性的单个过滤器参数,当该参数在变量中时,我的性能会变得杂草吗?对此我能做些什么吗?

它与查询中的分析功能有关吗?

这是我的查询:

CREATE FUNCTION fn_test()
RETURNS TABLE
WITH SCHEMABINDING
AS
    RETURN
    SELECT DISTINCT GCN_SEQNO, Drug_package_version_ID
    FROM
    (
        SELECT COALESCE(ndctbla.GCN_SEQNO, ndctblb.GCN_SEQNO) AS GCN_SEQNO,
            dpv.Drug_package_version_ID, ROW_NUMBER() OVER (PARTITION BY dpv.Drug_package_version_id ORDER BY 
                ndctbla.GCN_SEQNO DESC) AS Predicate
        FROM dbo.Drug_Package_Version dpv
            LEFT JOIN dbo.NDC ndctbla ON ndctbla.NDC = dpv.Sp_package_code
            LEFT JOIN dbo.NDC ndctblb ON ndctblb.SPC_NDC = dpv.Sp_package_code
    ) iq
    WHERE Predicate = 1
GO

GRANT SELECT ON fn_test TO public
GO

-- very fast
SELECT GCN_SEQNO
FROM dbo.fn_test()
WHERE Drug_package_version_id = 10000

GO

-- comparatively slow
DECLARE @dpvid int
SET @dpvid = 10000
SELECT GCN_SEQNO
FROM dbo.fn_test()
WHERE Drug_package_version_id = @dpvid
4

3 回答 3

2

通过 UDF 创建新投影后,不能期望您的索引仍将应用于在原始表上编制索引并包含在投影中的列。当您过滤投影(而不是在 UDF 中针对具有索引的原始表)时,索引不再适用。

你想要做的是参数化函数以接受参数。

如果您发现要设置参数的字段太多,那么您可能需要查看索引视图,因为您可以创建投影并对其进行索引,然后对其运行查询。

于 2009-04-02T16:18:30.293 回答
1

简单地说,常数很容易在计划中评估。局部变量不是。尤其是排序功能和过滤器Predicate = 1

解释 casparOne,您需要将过滤器尽可能向内推,以便dpv.Drug_package_version_id在 iq 派生表内部进行过滤。

如果你这样做,那么你也不需要 ,PARTITION BY因为你只有一个dpv.Drug_package_version_id. 然后你可以做一个清洁工...TOP 1 ... ORDER BY ndctbla.GCN_SEQNO DESC

于 2009-04-06T20:04:28.517 回答
1

我得到的回答很好,我也从中学习,但我想我找到了一个让我满意的答案。

我确实认为是使用 PARTITION BY 子句导致了这里的问题。我使用自连接习语的变体重新表述了 UDF:

SELECT t1.A, t1.B, t1.C
FROM T t1
    INNER JOIN
    (
        SELECT A, MAX(C) AS C
        FROM T
        GROUP BY A
    ) t2 ON t1.A = t2.A AND t1.C = t2.C

具有讽刺意味的是,这比使用特定于 SQL 2008 的查询的性能更高,而且优化器在使用变量而不是常量连接这个版本的查询时没有问题。在这一点上,我的结论是优化器只是不处理较新的 SQL 扩展以及较旧的东西。作为奖励,我现在可以在预升级的 SQL 2000 平台中使用 UDF。

谢谢大家的帮助!

于 2009-04-07T15:05:31.160 回答