簡體   English   中英

如何防止SQL服務器多次運行同一個子查詢

[英]How to prevent SQL Server from running the same subquery multiple times

我有一個遵循以下結構的查詢:

SELECT * 
FROM 
    ... <generated code> ...
    (SELECT <fields>,  
        CASE(SELECT TOP 1 ID FROM [Configuration] WHERE IsDefault=1 ORDER BY ID)
            WHEN 1 THEN t.FirstName
            WHEN 2 THEN t.LastName END As Identifier
     FROM <table> t) AS tmp
... <generated code> ...
WHERE <generated filters>

在查詢執行計划中,我看到Configuration表上的Clustered Index Scan的執行次數與<table>中的數字相同,但是,我知道這些掃描的結果總是相同的,當我更換

SELECT TOP 1 ID 
FROM [Configuration] 
WHERE IsDefault = 1 
ORDER BY ID

部分對於配置的當前值,此查詢運行速度很快。

我正在尋找一種方法來告訴 SQL 服務器這個子查詢總是有相同的結果以便它運行得很快,我看到的最明顯的方法是用該查詢的值聲明一個臨時變量並在主查詢中使用該變量,問題是查詢的開始和結束是由應用程序代碼生成的,我無法手動控制它。

對我來說,理想的解決方案是創建一個運行該查詢的確定性 function,並讓 SQL 服務器知道,由於 function 是確定性的,並且它不依賴於當前行,它只需要運行一次,但對於出於某種原因,它只是沒有用,它仍然運行了很多次。

go 我應該怎么優化這個? 我誤解了確定性函數嗎? 我剛才是不是把 function 弄錯了? 還有別的辦法嗎?

如評論中所述,使用CROSS JOIN有效,最終查詢為:

SELECT * 
FROM 
     ... <generated code> ...
    (SELECT <fields>,  
        CASE config.ID
            WHEN 1 THEN t.FirstName
            WHEN 2 THEN t.LastName END As Identifier
     FROM <table> t
     CROSS JOIN (SELECT TOP 1 ID FROM [Configuration] WHERE IsDefault=1 ORDER BY ID) config) AS tmp
... <generated code> ...
WHERE <generated filters>

正如問題中提到的,使用臨時變量會修復執行計划,但對我來說不是一個選項,因為查詢的頂部是由來自不同組件的代碼生成的

DECLARE @config INT
SET @config = (SELECT TOP 1 ID FROM [Configuration] WHERE IsDefault=1 ORDER BY ID)
SELECT * 
FROM 
     ... <generated code> ...
    (SELECT <fields>,  
        CASE @config
            WHEN 1 THEN t.FirstName
            WHEN 2 THEN t.LastName END As Identifier
     FROM <table> t) AS tmp
... <generated code> ...
WHERE <generated filters>

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM