譯:為什麼會從sys.dm_exec_query_plan中得到query_plan為NULL的結果

来源:http://www.cnblogs.com/wy123/archive/2017/12/03/7965651.html
-Advertisement-
Play Games

背景: 在面對生產環境的數據的異常問題診斷的時候,比如接到異常告警郵件,通常是CPU居高不下、長時間連續大批量發生物理IO(導致系統響應緩慢)、亦或是大量Session被阻塞或者大量session執行超出預期等等類似情況,筆者習慣性地利用sys.dm_exec_requests 和 sys.dm_e ...


 

背景:

在面對生產環境的數據的異常問題診斷的時候,比如接到異常告警郵件,
通常是CPU居高不下、長時間連續大批量發生物理IO(導致系統響應緩慢)、亦或是大量Session被阻塞或者大量session執行超出預期等等類似情況,
筆者習慣性地利用sys.dm_exec_requests 和 sys.dm_exec_sql_text(sql_handle) t 以及sys.dm_exec_query_plan(plan_handle) 三個系統表去觀察當前資料庫正在運行哪些Session,
活動的Session正在執行的批處理或者存儲過程是哪個?
正在執行批處理或者存儲過程中的哪一句sql?
活動的Session是否遇到到了阻塞?
活動的session的運行狀態(runnable或者是suspended)?
等待的資源是什麼?
sql語句是怎麼執行的(執行計劃,本文的重點要提及到的)?
以及在有參數的情況下,執行計劃編譯的參數等等信息。
獲取實時Session運行的sql語句的執行計劃是非常重要的參考信息之一,偶爾會發現一些正在執行的存儲過程的執行計劃為null(從sys.dm_exec_query_plan中查詢的),
這樣的話,得到的參考信息有限,不利於問題的診斷和解決。
參考下圖:

在網上發現一篇有對此問題分析的文章,覺的寫的很好,原文出處:https://blogs.msdn.microsoft.com/psssql/2016/07/13/why-am-i-getting-null-values-for-query_plan-from-sys-dm_exec_query_plan/

另外:
不僅僅在查詢緩存執行計劃的時候存在該問題,
sys.dm_exec_query_plan返回的是整個批處理中所有的語句的執行計劃,只要任何一個SQL語句的執行計劃沒有生成,返回的批處理的執行計劃就為null
對於實時運行的SQL(批處理或者存儲過程),如果是第一次運行或者是類似於作業任務執行頻率很低的sql(其執行計劃緩存被清理),
在真正運行之前的時候,
某些語句中存在臨時表的情況下,不會對批處理中所有的語句進行預編譯(也就是完整地編譯整個批處理),
因此利用sys.dm_exec_query_plan查詢的時候,只要批處理中有一句SQL未被編譯,都是無法獲取當前Session運行的批處理的執行計劃的。
對於這種情況,可以利用sys.dm_exec_text_query_plan 這個系統視圖,可以得到批處理中語句級的執行計劃。


 

以下為譯文:

 

最近我接到一個用戶打來的電話,想知道為什麼他會從 sys.dm_exec_query_plan中到NULL值的執行計劃,這個客戶參考了從 https://dzone.com/articles/dmexecqueryplan-returning-null參考了一篇本文,
在那個場景中,如果語句包含了臨時表,並且沒有被執行過,你會得到一個NULL之的執行計劃。
參考如下代碼:

SELECT plan_handle,usecounts, cacheobjtype, objtype, size_in_bytes, text, 
    qp.query_plan, tqp.query_plan AS text_query_plan
FROM sys.dm_exec_cached_plans cp
CROSS APPLY sys.dm_exec_sql_text(plan_handle) t
CROSS APPLY sys.dm_exec_query_plan(plan_handle) qp
CROSS APPLY sys.dm_exec_text_query_plan(plan_handle, NULL, NULL) tqp
WHERE text LIKE '%ConditionalPlanTest%'
AND objtype = 'Proc'

 

在執行調試併進入源代碼之後,發現了與這個問題更多的話題,為了弄清楚這個問題,我們需要瞭解幾個關鍵的概念

 

延遲編譯Deferred compilation

當第一次執行包含了多個語句的批處理或者存儲過程的時候,不是所有的語句都會被編譯。如果一些代碼分支導致一些語句不會被執行,SQL Server可能會就不會去編譯這些語句。換句話數,一部分語句只有第一次執行的時候才會被編譯,到目前為止,我發現,如果代碼分支被跳過的情況下,由如下兩種情況後導致延遲編譯。

1,語句包含臨時表。臨時表的數據可能會在實際執行的時候發生變化,因此,一開始就編譯這些sql語句是沒有意義的(譯註:臨時表的數據量,分佈等信息會影響到最終的執行計劃)
2,語句附加了option (recompile)選項,這種情況下每次都會重新編譯語句,如果不使用它,為什麼要提前編譯它?

 

你想要的是整個批處理的還是語句級的執行計劃

sys.dm_exec_query_plan提供了整個批處理(或者存儲過程)的執行計劃,而sys.dm_exec_text_query_plan 允許你根據語句偏移開始位置和結束位置(statement_start_offset and statement_end_offset)獲取語句級的執行計劃,當然, sys.dm_exec_text_query_plan在沒有提供語句偏移量的時候,可以返回整個批處理的執行計劃

為什麼是NULL值

當指定獲取整個批處理(或者存儲過程)的執行計劃的時候,SQL Server將會檢索整個批處理或者存儲過程中的所有語句的執行計劃,如果任何一個語句的執行計劃不存在,那麼就是返回NULL值,因為執行計劃是不完整的(對於批處理或者存儲過程來說)。註意,上述獲取執行計劃的語句是從sys.dm_exec_query_plan中查詢的,意味著需要返回整個批處理的執行計劃,這就是為什麼返回NULL值的原因。

在一個不是所有語句都被編譯的批處理中,如何獲取語句級別的執行計劃

你可以在“提供批處理統計”的情況下修改上述查詢語句,從sys.dm_exec_sql_text中可以得到哪些因為包含了臨時表或者 option (recompile)的導致沒有提前完整編譯的語句級執行計劃,這裡有一個查詢示例,請註意需要提供嚴格的語句偏移開始和結束位置信息

SELECT s2.dbid,   
    s1.sql_handle,    
    (SELECT TOP 1 SUBSTRING(s2.text,statement_start_offset / 2+1 ,   
      ( (CASE WHEN statement_end_offset = -1   
         THEN (LEN(CONVERT(nvarchar(max),s2.text)) * 2)   
         ELSE statement_end_offset END)  - statement_start_offset) / 2+1))  AS sql_statement,  
    execution_count,   
    plan_generation_num,   
    last_execution_time,     
    total_worker_time,   
    last_worker_time,   
    min_worker_time,   
    max_worker_time,  
    total_physical_reads,   
    last_physical_reads,   
    min_physical_reads,    
    max_physical_reads,    
    total_logical_writes,   
    last_logical_writes,   
    min_logical_writes,   
    max_logical_writes    ,
                cast(s3.query_plan as xml) query_plan
FROM sys.dm_exec_query_stats AS s1   
CROSS APPLY sys.dm_exec_sql_text(sql_handle) AS s2    
cross apply sys.dm_exec_text_query_plan (plan_handle, statement_start_offset, statement_end_offset) s3
where object_name ( s2.objectid, s2.dbid) = 'ConditionalPlanTest'
ORDER BY s1.sql_handle, s1.statement_start_offset, s1.statement_end_offset;  

DEMO

譯註:存儲過程中因為代碼分支邏輯,導致一部分代碼無法執行,無法執行的代碼分支也就沒有編譯,試圖用sys.dm_exec_query_plan返回執行計劃的時候,因為缺少一部分代碼的執行計劃,返回的執行計劃欄位為空

use tempdb
go

create table t1 (c1 int)
go
create table t2 (c1 int)
go

create procedure p_test @option int
as
if @option >= 2
select * from t1 option (recompile)
if @option >=1
select * from t2 option (recompile)

go
--this will only execute  2nd statement and skip first statement
p_test 1
go

--NULL plan will be returned because the query wants whole batch plan
SELECT plan_handle,usecounts, cacheobjtype, objtype, size_in_bytes, text, 
    s3.query_plan AS text_query_plan
FROM sys.dm_exec_cached_plans s1
CROSS APPLY sys.dm_exec_sql_text(plan_handle) s2
CROSS APPLY sys.dm_exec_query_plan(plan_handle) s3
WHERE object_name ( s2.objectid, s2.dbid) = 'p_test'
因為試圖返回整個批處理的執行計劃,將會返回空的執行計劃
image
用sys.dm_exec_text_query_plan 返回執行計劃的時候,可以返回批處理中語句級的執行計劃
--this will return the plan for 2nd statement that was executed
--but it will not return plan for 1st statement because the query was never executed and compilation was deferred
SELECT s2.dbid,   
    s1.sql_handle,    
    (SELECT TOP 1 SUBSTRING(s2.text,statement_start_offset / 2+1 ,   
      ( (CASE WHEN statement_end_offset = -1   
         THEN (LEN(CONVERT(nvarchar(max),s2.text)) * 2)   
         ELSE statement_end_offset END)  - statement_start_offset) / 2+1))  AS sql_statement,  
         cast(s3.query_plan as xml) query_plan,
    execution_count,   
    plan_generation_num,   
    last_execution_time,     
    total_worker_time,   
    last_worker_time,   
    min_worker_time,   
    max_worker_time,  
    total_physical_reads,   
    last_physical_reads,   
    min_physical_reads,    
    max_physical_reads,    
    total_logical_writes,   
    last_logical_writes,   
    min_logical_writes,   
    max_logical_writes    
                
FROM sys.dm_exec_query_stats AS s1   
CROSS APPLY sys.dm_exec_sql_text(sql_handle) AS s2    
cross apply sys.dm_exec_text_query_plan (plan_handle, statement_start_offset, statement_end_offset) s3
where object_name ( s2.objectid, s2.dbid) = 'p_test'
ORDER BY s1.sql_handle, s1.statement_start_offset, s1.statement_end_offset; 

 

 

 

 

插曲:

中間遇到一個悲劇的問題:基本上寫完的情況下,往文章裡面粘代碼,按下ctrl+v之後,瞬間就變成這種樣子了,因為不是從草稿狀態編輯的,整篇文章都沒有了,隨時可以重現。

再刷新,內容裡面毛線都沒有了,┗|`O′|┛ 嗷~~

 

 



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 列表標簽 有序列表: <ol><li></li></ol> 無序列表: <ul><li></li></ul> //屬性:font-weight字體粗細 表格: <table> <thead><tr><td></td></tr></thead> //tr行 td列 <tbody><tr><td></t ...
  • 1、尊重對象所有權  不要為實例或原型添加屬性; 不要為實例或原型添加方法; 不要重定義已存在的方法。 最佳的方法便是永遠不修改不是由你所有的對象。所謂擁有對象,就是說這個對象是你創建的,比如你自己創建的自定義類型或對象字面量。而 Array 、 document 這些顯然不是你的,它們在你的 ...
  • 1.正則表達式test方法 test() 方法用於檢測一個字元串是否匹配某個模式 返回值: 如果字元串 string 中含有與 RegExpObject 匹配的文本,則返回 true,否則返回 false。 2、正則表達式exec方法 exec() 方法用於檢索字元串中的正則表達式的匹配。 返回值: ...
  • 1、match方法 match() 方法可在字元串內檢索指定的值,或找到一個或多個正則表達式的匹配。 match()方法的返回值為:存放匹配結果的數組。 2、replace方法 replace() 方法用於在字元串中用一些字元替換另一些字元,或替換一個與正則表達式匹配的子串。 replace方法的返 ...
  • 在knockoutjs 上實現 Flux 單向數據流 狀態機,主要解決多個組件之間對數據的耦合問題。 一、其實簡單 flux的設計理念和實現方案,很大程度上人借鑒和參考了Vuex的實現,只是簡化了某些過程,數據流向圖如下:從上圖,中以看出數據的改變是單向迴圈的。我想這就是Flux理念的核心所在吧。V ...
  • vue cli vue在web前端可謂是大放異彩,尤其在國內與angular、react有三足鼎立之勢。很多人想入門vue2而又苦於不知從何下手。因為vue2是以組件化開發的,最好要搭配webpack構建工具開發,而webpack很多人還不能獨立配置。 而vue cli腳手架就能很好解決這一問題。即 ...
  • 第一種:傳統的ajax非同步請求,後臺代碼以及效果在最下邊 首先我們在eclipse中創建一個註冊頁面regist.jsp,創建一個form表單,註意,由於我們只是實現用戶名校驗的效果,下邊紅色部門是我們需要研究對象,所以其他的部門可以忽略不看。 內容如下: <%@ page language="ja ...
  • 1.indexof方法 indexOf() 方法可返回某個指定的字元串值在字元串中首次出現的位置。 語法: 註意:有可選的參數(即設置開始的檢索位置)。 2、search方法 search() 方法用於檢索字元串中指定的子字元串,或檢索與正則表達式相匹配的子字元串。 註意:search方法可以根據正 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...