[size=13.3333px]数据库引擎接收到一个新的查询请求(Batch或SP),查询优化器会生成执行计划,并缓存到内存中;下次再次执行相同的查询请求时,数据库引擎从复用已经缓存的执行计划,换句话,数据库引擎为每一个查询请求生成执行计划,并把已经生成的执行计划缓存起来,当接收到相同的查询请求时,数据库引擎复用已缓存的执行计划。查询请求(Batch或SP)中的每一个查询语句的执行计划,都会被缓存到内存中,数据库引擎统计执行计划的性能参数,缓存在DMV:sys.dm_exec_query_stats中,在该视图中,每一行数据都表示一个查询语句的统计数据: - sql_handle:用以唯一标识一个TSQL文本(Batch或SP),TSQL文本存储在SQL Manager Cache(SQLMGR)中;
- plan_handle:用于唯一标识一个已编辑的查询计划,查询计划存储在计划缓存中;
[size=13.3333px]一个sql_handle 能够生成多个查询计划,对应多个plan_handle,但是每个plan_handle只能对应一个sql_handle 。 [size=13.3333px]一,获取查询语句 [size=13.3333px]视图sys.dm_exec_query_stats 缓存的是单个查询语句的执行计划,而sql_handler引用的是整个TSQL文本(Batch或SP),为了获得单个查询语句的文本,必须通过语句的偏移字段来抽取,偏移量是字节,字节数量从0开始: - statement_start_offset:语句开始偏移的字节序号
- statement_end_offset:语句结束偏移的字节序号,-1 表示TSQL文本的末尾;
[size=13.3333px]由于函数 sys.dm_exec_sql_text 返回的TSQL文本是以nvarchar(max)类型存储的,一般情况下,字节偏移量都是2的倍数,获取查询语句的脚本是: select substring(st.text ,qs.statement_start_offset/2+1, ( case when qs.statement_end_offset = -1 then len(convert(nvarchar(max), st.text)) else (qs.statement_end_offset - qs.statement_start_offset)/2 end ) ) as individual_query ,st.text as entire_query from sys.dm_exec_query_stats qs outer apply sys.dm_exec_sql_text(qs.sql_handle) as st
|