如果您希望通过sys.dm_exec_requests.plan_handle 在sys.dm_exec_sql_text 中看到exec dbo.sp_download_files_mail,那么这可能不会发生。当您使用EXEC 时,它会创建一个可能是不同执行计划的子流程。在 SQLCLR 的情况下,除非您使用 SqlConnection 执行 T-SQL,否则 SQL Server 无法洞察正在发生的事情,然后您将获得在 SQLCLR 对象中执行的 SQL 的计划,而不是 SQLCLR 的计划对象本身。当你没有执行任何T-SQL语句时,sql_handle和plan_handle值为空: 0x000000000000/300000000/3000000//dr>
但是,您可以看到 SQLCLR 对象显示在从 sys.dm_exec_cached_plans 返回的 plan_handle 值中。 SQLCLR 对象在执行时似乎确实出现在此 DMV 中,但由于此 DMV 报告缓存的对象,它不一定会在 SQLCLR 对象完成后被删除。因此,您不能使用此 DMV 来指示对象的当前运行状态。 sys.dm_exec_cached_plans 中报告的 plan_handle 值也不会在运行时显示在 sys.dm_exec_requests 中。
您可以自己测试此行为,方法是创建一个 SQLCLR 存储过程或 SQLCLR 标量用户定义函数,它只会调用 System.Threading.Thread.Sleep() 至少 30 秒。如果您不想创建这个,一个预制的 SQLCLR UDF - DB_WaitForDelay - 存在于SQL# SQLCLR 库(我创建的)的免费版本中,并且是我在下面的示例代码。
在 SQL Server Management Studio (SSMS) 中,打开两个查询选项卡并粘贴以下内容:
标签 1
EXEC [SQL#].[DB_WaitForDelay] 30000, 1;
标签 2
SELECT txt.*, req.*
FROM sys.dm_exec_requests req
OUTER APPLY sys.dm_exec_sql_text(req.[plan_handle]) txt
WHERE req.[session_id] = <session_id_of_Tab1>;
DBCC INPUTBUFFER(<session_id_of_Tab1>);
SELECT txt.*, cp.*
FROM sys.dm_exec_cached_plans cp
OUTER APPLY sys.dm_exec_sql_text(cp.[plan_handle]) txt
WHERE cp.[cacheobjtype] LIKE N'CLR%';
在 Tab 2 查询中替换“<session_id_of_Tab1>”的两个实例后,执行 Tab 1 查询,然后返回 Tab 2 并执行这批查询。
如果您确实需要知道此 SQLCLR 对象是否在执行时在执行时,那么您将不得不按照使用SqlConnection 和Context Connection = true; 的ConnectionString 和然后执行(在 SQLCLR 对象的开头)类似SET CONTEXT_INFO 0x1234; 的内容(假设您尚未将CONTEXT_INFO 用于其他内容)。在 SQLCLR 对象的末尾,为SET CONTEXT_INFO 0x00; 执行第二个SqlCommand 以将其清除。
这种方法允许您使用以下查询来确认它当前正在运行:
SELECT req.*
FROM sys.dm_exec_requests req
WHERE req.[context_info] = 0x1234;
此外,在存储过程名称前加上 sp_ 是一种相当糟糕的做法,因为这会导致 SQL Server 首先检查 [master] 的对象,然后然后检查当前数据库。使用spDownloadEmailFiles 之类的东西会更好,尽管仍然没有真正好的理由为存储过程/函数/表/视图名称添加任何前缀。