cjb*_*rth 35 sql-server stored-procedures sql-server-2008
我正在使用SQL Server 2008中的存储过程,并且我已经知道我必须INSERT INTO
使用已预定义的临时表来处理数据.这很好,除了如何定义我的临时表,如果我不是那个编写存储过程而不是列出其定义和读取代码的人?
例如,我的临时表对于'EXEC sp_stored_procedure'是什么样的?这是一个简单的存储过程,我可能猜测数据类型,但似乎必须有一种方法来只读取执行过程返回的列的类型和长度.
Aar*_*and 53
所以假设你在tempdb中有一个存储过程:
USE tempdb;
GO
CREATE PROCEDURE dbo.my_procedure
AS
BEGIN
SET NOCOUNT ON;
SELECT foo = 1, bar = 'tooth';
END
GO
Run Code Online (Sandbox Code Playgroud)
有一种非常复杂的方法可以确定存储过程将输出的元数据.有几个注意事项,包括该过程只能输出单个结果集,如果无法精确确定数据类型,则会对数据类型进行最佳猜测.它需要使用OPENQUERY
和环回链接的服务器,并将'DATA ACCESS'
属性设置为true.您可以检查sys.servers以查看您是否已经拥有有效的服务器,但是我们只需创建一个手动调用的服务器loopback
:
EXEC master..sp_addlinkedserver
@server = 'loopback',
@srvproduct = '',
@provider = 'SQLNCLI',
@datasrc = @@SERVERNAME;
EXEC master..sp_serveroption
@server = 'loopback',
@optname = 'DATA ACCESS',
@optvalue = 'TRUE';
Run Code Online (Sandbox Code Playgroud)
现在您可以将其作为链接服务器进行查询,您可以将任何查询的结果(包括存储过程调用)用作常规SELECT
.所以你可以这样做(注意数据库前缀很重要,否则你会得到错误11529和2812):
SELECT * FROM OPENQUERY(loopback, 'EXEC tempdb.dbo.my_procedure;');
Run Code Online (Sandbox Code Playgroud)
如果我们可以执行a SELECT *
,我们还可以执行SELECT * INTO
:
SELECT * INTO #tmp FROM OPENQUERY(loopback, 'EXEC tempdb.dbo.my_procedure;');
Run Code Online (Sandbox Code Playgroud)
一旦#tmp表存在,我们可以通过说(假设SQL Server 2005或更高版本)来确定元数据:
SELECT c.name, [type] = t.name, c.max_length, c.[precision], c.scale
FROM sys.columns AS c
INNER JOIN sys.types AS t
ON c.system_type_id = t.system_type_id
AND c.user_type_id = t.user_type_id
WHERE c.[object_id] = OBJECT_ID('tempdb..#tmp');
Run Code Online (Sandbox Code Playgroud)
(如果您使用的是SQL Server 2000,则可以使用syscolumns执行类似操作,但我没有2000实例来验证等效查询.)
结果:
name type max_length precision scale
--------- ------- ---------- --------- -----
foo int 4 10 0
bar varchar 5 0 0
Run Code Online (Sandbox Code Playgroud)
在Denali,这将变得更加容易.同样,第一个结果集仍然存在限制,但您不必设置链接服务器并跳过所有这些环节.你可以说:
DECLARE @sql NVARCHAR(MAX) = N'EXEC tempdb.dbo.my_procedure;';
SELECT name, system_type_name
FROM sys.dm_exec_describe_first_result_set(@sql, NULL, 1);
Run Code Online (Sandbox Code Playgroud)
结果:
name system_type_name
--------- ----------------
foo int
bar varchar(5)
Run Code Online (Sandbox Code Playgroud)
直到Denali,我建议你可以更容易地卷起袖子并自己找出数据类型.这不仅仅是因为完成上述步骤很繁琐,而且还因为您更有可能做出比引擎更正确(或至少更准确)的猜测,因为数据类型猜测引擎所做的将基于运行时输出,没有任何可能值的域的外部知识.这个因素在Denali中也是如此,所以不要觉得新的元数据发现功能是最终的结果,它们只会使上面的内容变得不那么乏味.
哦,对于其他一些潜在的问题OPENQUERY
,请参阅Erland Sommarskog的文章:
http://www.sommarskog.se/share_data.html#OPENQUERY
一种不太复杂的方法(在某些情况下可能就足够了):编辑原始SP,在最终SELECT之后和FROM子句之前添加INSERT INTO tmpTable以将SP结果保存在tmpTable中.
运行修改后的SP,最好使用有意义的参数,以获得实际数据.恢复该过程的原始代码.
现在,您可以从SQL Server management studio获取tmpTable脚本或查询sys.columns以获取字段描述.
看起来在SQL 2012中有一个新的SP来帮助解决这个问题.
exec sp_describe_first_result_set N'PROC_NAME'
Run Code Online (Sandbox Code Playgroud)
小智 6
这是我写的一些代码.这个想法(正如其他人所说)是获取SP代码,修改它并执行它.但是,我的代码不会更改原始SP.
第一步,获取SP的定义,剥离"创建"部分,并在声明参数后删除"AS"(如果存在).
Declare @SPName varchar(250)
Set nocount on
Declare @SQL Varchar(max), @SQLReverse Varchar(MAX), @StartPos int, @LastParameterName varchar(250) = '', @TableName varchar(36) = 'A' + REPLACE(CONVERT(varchar(36), NewID()), '-', '')
Select * INTO #Temp from INFORMATION_SCHEMA.PARAMETERS where SPECIFIC_NAME = 'ADMIN_Sync_CompareDataForSync'
if @@ROWCOUNT > 0
BEGIN
Select @SQL = REPLACE(ROUTINE_DEFINITION, 'CREATE PROCEDURE [' + ROUTINE_SCHEMA + '].[' + ROUTINE_NAME + ']', 'Declare')
from INFORMATION_SCHEMA.ROUTINES
where ROUTINE_NAME = @SPName
Select @LastParameterName = PARAMETER_NAME + ' ' + DATA_TYPE +
CASE WHEN CHARACTER_MAXIMUM_LENGTH is not null THEN '(' +
CASE WHEN CHARACTER_MAXIMUM_LENGTH = -1 THEN 'MAX' ELSE CONVERT(varchar,CHARACTER_MAXIMUM_LENGTH) END + ')' ELSE '' END
from #Temp
WHERE ORDINAL_POSITION =
(Select MAX(ORDINAL_POSITION)
From #Temp)
Select @StartPos = CHARINDEX(@LastParameterName, REPLACE(@SQL, ' ', ' '), 1) + LEN(@LastParameterName)
END
else
Select @SQL = REPLACE(ROUTINE_DEFINITION, 'CREATE PROCEDURE [' + ROUTINE_SCHEMA + '].[' + ROUTINE_NAME + ']', '') from INFORMATION_SCHEMA.ROUTINES where ROUTINE_NAME = @SPName
DROP TABLE #Temp
Select @StartPos = CHARINDEX('AS', UPPER(@SQL), @StartPos)
Select @SQL = STUFF(@SQL, @StartPos, 2, '')
Run Code Online (Sandbox Code Playgroud)
(注意基于唯一标识符创建新表名)现在找到代码中的最后一个'From'字,假设这是执行返回结果集的select的代码.
Select @SQLReverse = REVERSE(@SQL)
Select @StartPos = CHARINDEX('MORF', UPPER(@SQLReverse), 1)
Run Code Online (Sandbox Code Playgroud)
更改代码以将结果集选择到表中(基于uniqueidentifier的表)
Select @StartPos = LEN(@SQL) - @StartPos - 2
Select @SQL = STUFF(@SQL, @StartPos, 5, ' INTO ' + @TableName + ' FROM ')
EXEC (@SQL)
Run Code Online (Sandbox Code Playgroud)
结果集现在在一个表中,表是否为空无关紧要!
让我们得到表的结构
Select * from INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = @TableName
Run Code Online (Sandbox Code Playgroud)
你现在可以做到这一点
别忘了丢掉那张独特的桌子
Select @SQL = 'drop table ' + @TableName
Exec (@SQL)
Run Code Online (Sandbox Code Playgroud)
希望这可以帮助!