共用方式為


移轉後驗證和最佳化指南

適用於:SQL Server

SQL Server 移轉後步驟對於協調任何資料精確度和完整性,以及發現工作負載的效能問題至關重要。

常見效能案例

以下是一些移轉至 SQL Server 平台後常發生的效能案例以及解決方法。 這些包含 SQL Server 至 SQL Server 移轉的特定案例 (舊版移轉至新版),以及外部平台 (例如 Oracle、DB2、MySQL 及 Sybase) 移轉至 SQL Server。

因為基數估算器 (CE) 版本變更造成的查詢衰退

適用於:SQL Server 至 SQL Server 的移轉。

從舊版 SQL Server 移轉至 SQL Server 2014 (12.x) 或更新版本,並升級到最新的資料庫相容性層級時,工作負載可能會有效能迴歸的風險。

這是因為從 SQL Server 2014 (12.x) 開始,所有的查詢最佳化工具變更都會繫結至最新的資料庫相容性層級;因此,計劃不會在升級時立即變更,而是在使用者將 COMPATIBILITY_LEVEL 資料庫選項變更為最新版本時變更。 此功能會結合查詢存放區,可讓您在升級過程中對查詢效能擁有絕佳層級的控制。

如需 SQL Server 2014 (12.x) 所導入的查詢最佳化工具變更的詳細資訊,請參閱使用 SQL Server 2014 基數估算程式最佳化您的查詢計劃

如需有關 CE 的詳細資訊,請參閱基數估計 (SQL Server)

解決步驟

資料庫相容性層級變更成來源版本並遵循建議的升級工作流程,如下圖所示:

 顯示建議升級工作流程的圖表。

如需本文的詳細資訊,請參閱在升級到新版 SQL Server 期間保持效能穩定

參數探查的敏感度

適用於:外部平台 (例如 Oracle、DB2、MySQL 及 Sybase) 至 SQL Server 的移轉。

注意

針對 SQL Server 至 SQL Server 的移轉,若此問題存在於來源 SQL Server 中,依現況移轉至較新版本的 SQL Server 將無法處理此種狀況。

SQL Server 透過在第一次編譯時使用探查輸入參數來編譯預存程序的查詢計劃,產生已針對該輸入資料分佈進行最佳化的參數化和可重複使用的計畫。 即使不是預存程序,也會將產生簡單計劃的大部分陳述式進行參數化。 第一次快取計劃之後,日後的每次執行都會對應至先前快取的計劃。

第一次編譯時若未對一般工作負載使用最常見的參數集,就會引發潛在問題。 對於不同的參數,使用相同的執行計畫會變成效率不佳。 如需本文的詳細資訊,請參閱參數敏感度

解決步驟

  1. 使用 RECOMPILE 提示。 每次調整每個參數值時,計劃會計算一次。
  2. 重新撰寫預存程序來使用 (OPTIMIZE FOR(<input parameter> = <value>)) 選項。 決定要使用哪個適合大部分相關工作負載的值,建立和維護一個對於參數化值而言變得有效率的計劃。
  3. 使用程序內的區域變數重新撰寫預存程序。 現在,最佳化工具會使用密度向量進行估計,導致不論參數值為何,都會產生相同的計劃。
  4. 重新撰寫預存程序來使用 (OPTIMIZE FOR UNKNOWN) 選項。 其效果與使用區域變數技巧相同。
  5. 重新撰寫查詢來使用 DISABLE_PARAMETER_SNIFFING 提示。 除非使用 OPTION(RECOMPILE)WITH RECOMPILEOPTIMIZE FOR <value>,否則完全停用參數探查,其效果與使用區域變數技巧相同。

提示

使用 Management Studio 計劃分析功能來快速確認這是否是問題。 如需詳細資訊,請參閱 SSMS 的新功能:查詢效能疑難排解變得更簡單!

遺漏索引

適用於:外部平台 (例如 Oracle、DB2、MySQL 及 Sybase) 和 SQL Server 至 SQL Server 的移轉。

索引不正確或遺漏會造成額外的 I/O,而導致額外的記憶體和 CPU 浪費。 這可能是因為工作負載設定檔已變更,例如使用不同的述詞,而使現有索引設計失效。 索引策略不佳或工作負載設定檔變更的辨識項包括:

  • 尋找重複、多餘、很少使用和完全未使用的索引。
  • 特別注意未使用的索引與更新。

解決步驟

  1. 將圖形化執行計畫用於任何缺少的索引參考。
  2. 索引建議由 Database Engine Tuning Advisor 產生。
  3. 使用 sys.dm_db_missing_index_details 或透過 SQL Server 效能儀表板
  4. 使用既有的指令碼,可以使用現有的 DMV 深入了解任何遺漏、重複、多餘、很少使用和完全未使用的索引,以及是否有任何索引參考已提示/硬式編碼成資料庫中的現有程序和函數。

提示

這類既有指令碼的範例包括 索引建立索引資訊

無法使用述詞來篩選資料

適用於:外部平台 (例如 Oracle、DB2、MySQL 及 Sybase) 和 SQL Server 至 SQL Server 的移轉。

注意

針對 SQL Server 至 SQL Server 的移轉,若此問題存在於來源 SQL Server 中,依現況移轉至較新版本的 SQL Server 將無法處理此種狀況。

SQL Server 查詢最佳化工具只能說明編譯時已知的資訊。 如果工作負載依賴只有在執行時才能得知的述詞,則會提高選擇到不佳計畫的可能性。 在品質更高的計劃中,述詞必須是 SARGableSearch Argumentable

非 SARGable 述詞的一些範例:

  • 隱含資料轉換,如 varchar 轉換為 nvarchar,或 int 轉換為 varchar。 在實際執行計畫中尋找執行階段 CONVERT_IMPLICIT 警告。 從某個類型轉換成另一個類型也可能造成致遺失有效位數。
  • 複雜的不明運算式,例如 WHERE UnitPrice + 1 < 3.975,但不是 WHERE UnitPrice < 320 * 200 * 32
  • 使用函數的運算式,例如 WHERE ABS(ProductID) = 771WHERE UPPER(LastName) = 'Smith'
  • 包含前置萬用字元的字串,例如 WHERE LastName LIKE '%Smith',但不是 WHERE LastName LIKE 'Smith%'

解決步驟

  1. 一律將變數/參數宣告為預期的目標資料型別

    這可能涉及將資料庫所儲存的任何使用者定義程式碼建構 (例如預存程序、使用者定義函數或檢視表),與保存基礎表格所用資料類型相關資訊的系統資料表 (例如 sys.columns (Transact-SQL)) 進行比較。

  2. 如果無法周遊所有程式碼到上一個點,則基於相同目的,請變更資料表的資料類型以符合任何變數/參數宣告。

  3. 推斷出下列結構的效益:

    • 作為述詞使用的函數;
    • 萬用字元搜尋;
    • 根據單欄式資料的複雜運算式 - 評估是否需要改為建立可建立索引的保存計算資料行;

注意

所有這些步驟皆可以程式設計方式完成。

使用資料表值函式 (多重陳述式與內嵌)

適用於:外部平台 (例如 Oracle、DB2、MySQL 及 Sybase) 和 SQL Server 至 SQL Server 的移轉。

注意

針對 SQL Server 至 SQL Server 的移轉,若此問題存在於來源 SQL Server 中,依現況移轉至較新版本的 SQL Server 將無法處理此種狀況。

資料表值函數會傳回可以代替檢視的資料表資料類型。 檢視只限於單一 SELECT 陳述式,而使用者自訂函數可以包含其他陳述式,所允許的邏輯比在檢視中的還多。

重要

因為多重陳述式資料表值函式 (MSTVF) 的輸出資料表不會在編譯時建立,SQL Server 查詢最佳化工具會依賴啟發學習法,而不是實際的統計資料,來判斷資料列的估計值。 即使將索引加入基底資料表,也沒有什麼幫助。 針對 MSTVF,SQL Server 使用 1 的固定估計作為 MSTVF 預期傳回的資料列數目 (從 SQL Server 2014 (12.x) 開始,其固定估計為 100 個資料列)。

解決步驟

  1. 如果 MSTVF 僅為單一陳述式,請轉換為內嵌資料表值函式。

    CREATE FUNCTION dbo.tfnGetRecentAddress(@ID int)
    RETURNS @tblAddress TABLE
    ([Address] VARCHAR(60) NOT NULL)
    AS
    BEGIN
      INSERT INTO @tblAddress ([Address])
      SELECT TOP 1 [AddressLine1]
      FROM [Person].[Address]
      WHERE  AddressID = @ID
      ORDER BY [ModifiedDate] DESC
    RETURN
    END
    

    內嵌格式範例會顯示在下方。

    CREATE FUNCTION dbo.tfnGetRecentAddress_inline(@ID int)
    RETURNS TABLE
    AS
    RETURN (
      SELECT TOP 1 [AddressLine1] AS [Address]
      FROM [Person].[Address]
      WHERE  AddressID = @ID
      ORDER BY [ModifiedDate] DESC
    )
    
  2. 如果更加複雜,請考慮使用記憶體最佳化資料表或暫存資料表所儲存的中繼結果。