sp_MSforeachtable - 解析动态sql

Posted

技术标签:

【中文标题】sp_MSforeachtable - 解析动态sql【英文标题】:sp_MSforeachtable - parsing of dynamic sql 【发布时间】:2012-04-23 13:43:53 【问题描述】:

我最近发现了一个问题,我想使用 sp_MSforeachtable 存储过程来选择表名中包含 Transcode 一词的所有表,并在这些表上运行一些 SQL。我设法编写了一些有效的代码,但并不完美 - 对于那些我希望它能够优雅地跳过的表(即那些名称中没有转码的表),它反而会由于某些预期的列而引发错误(仅存在于转码表中)不存在于这些表中。问题似乎是在调用存储过程时会解析所有 SQL,而不是仅在需要时(例如满足条件时)解析 SQL。

以下代码按预期工作:

exec sp_MSforeachtable '
print ''Table being tested: ?''
if exists (select 1 where ''?'' like ''%Transcode%'')
begin
    print ''    Do Something''
end
else
begin
    print ''    Ignored''
end
'

但是,当我尝试添加功能时,我从永远不会运行的代码中得到错误;例如

exec sp_MSforeachtable '
print ''Table being tested: ?''
if exists (select 1 where ''?'' like ''%Transcode%'')
begin
    print ''    Do Something''
    
    insert ? (col1, col2, col3)
    select col1, col2, 1
    from ?
    where col3 = 0
    
end
else
begin
    print ''    Ignored''
end
'

这一次,对于那些表名包含单词 Transcode 的输出,我得到与第一个相同的输出,但对于那些不包含 Transcode 的输出,我看到:

消息 207,第 16 级,状态 1,第 9 行

列名 col3 无效

我很确定这取决于动态 SQL 的解析方式,但这是不受欢迎的行为。有没有人遇到过这种情况/是否有简单的解决方法?

这并不紧急,因为在我的情况下,由于列不存在,错误与 if 语句具有相同的效果,并且有效行能够成功运行,但我很想学习以防万一我需要尽快做类似的事情,这种行为会导致问题。

提前致谢,

JB

ps。复制此行为的代码如下:

create table DemoTranscode1 (id bigint identity(1,1) primary key clustered, col1 nvarchar(10) not null, col2 nvarchar(10)not null, col3 bit not null)
go
create table DemoTable1 (id bigint identity(1,1) primary key clustered, col1 nvarchar(10) not null, col2 nvarchar(10)not null)
go
create table DemoTranscode2 (id bigint identity(1,1) primary key clustered, col1 nvarchar(10) not null, col2 nvarchar(10)not null, col3 bit not null)
go
create table DemoTranscode3 (id bigint identity(1,1) primary key clustered, col1 nvarchar(10) not null, col2 nvarchar(10)not null, col3 bit not null)
go
insert DemoTranscode1
select 'example1', 'demo', 0
union select 'example2', 'demo', 0
union select 'example3', 'demo', 0
union select 'example4', 'demo', 0
insert DemoTable1 select col1, col2 from DemoTranscode1
insert DemoTranscode2 select col1, col2, col3 from DemoTranscode1
insert DemoTranscode3 select col1, col2, col3 from DemoTranscode1

【问题讨论】:

【参考方案1】:

一方面,我建议远离像sp_MSForEachTable 这样的无证和不受支持的程序。它们可以随时更改甚至从 SQL Server 中删除,并且此特定过程可能具有许多针对sp_MSForEachDb 报告的相同症状。 (查看一些背景 here 和 here。)

我会这样做:

DECLARE @sql NVARCHAR(MAX);
SELECT @sql = N'';

SELECT @sql = @sql + 'INSERT ' 
  + QUOTENAME(SCHEMA_NAME([schema_id]))
  + '.' + QUOTENAME(name) + ' (col1, col2, col3)
  SELECT col1, col2, 1 FROM '
  + QUOTENAME(SCHEMA_NAME([schema_id]))
  + '.' + QUOTENAME(name)
  + ' WHERE col3 = 0;'
FROM sys.tables 
WHERE name LIKE '%Transcode%';

PRINT @sql;
-- EXEC sp_executesql @sql;

这样做的好处是在执行之前很容易验证输出。

【讨论】:

太好了,谢谢 Aaron - 简单多了:)。附言。我还找到了另一个选项 - 嵌套动态 sql(Mikael Eriksson 在这里的回答:***.com/questions/9679997/…) - 但正如您所指出的,如果不支持 proc,最好不要使用。再次感谢,JB【参考方案2】:

您可以使用@whereand 参数,这样您就不需要签入您的代码。

exec sp_MSforeachtable
  @Command1 = 'print "?"',
  @whereand = ' and o.name like ''%Transcode%'''

更新:

我很确定这取决于动态 SQL 的解析方式,但是 这是不受欢迎的行为。有没有人遇到过这种情况

当然。代码在执行之前进行编译,编译器会根据表检查插入语句中使用的列名。

【讨论】:

谢谢米凯尔;没有依赖其他参数 - 我现在查看了这个存储过程的代码,所以对幕后发生的事情了解得不多。【参考方案3】:

问题在于解析器。无论您是否在某些条件下使用 sp_msforeachtable,它仍然会解析每个表。因此,对于其他表 - 引发错误。您可以使用 exec 语句来避免它,如下所示 -

exec sp_MSforeachtable ' 
print ''Table being tested: ?'' 
if exists (select 1 where ''?'' like ''%Transcode%'') 
begin 
    print ''    Do Something'' 

    exec ( ''insert ? (col1, col2, col3) select col1, col2, 1 from ? where col3 = 0 '') 

end 
else 
begin 
    print ''    Ignored'' 
end 
' 

【讨论】:

作为一个警告,我会小心insert ?之类的东西——这只是表名,如果架构不是dbo怎么办?如果表名为a tablemy-table 怎么办? 感谢两位cmets。 Aaron - 关于模式和特殊字符,我在插入插入语句之前做了检查,看看我是否应该附加模式/使用 quotename 函数 - 存储的过程返回?形式为 [dbo].[my-table];所以工作正常。同样在这种情况下,我知道其他模式中不会有 Transcode 表;但对于任何关注该主题的人来说都是一个有效的观点。 是的。 ?替换为 [schema_name].[table_name] 之类的内容。因此,如果 John 必须具体说明,请将语句替换为 - if exists (select 1 where ''?'' = ''[dbo].[Transcode]'')

以上是关于sp_MSforeachtable - 解析动态sql的主要内容,如果未能解决你的问题,请参考以下文章

为 sp_MSforeachtable 执行的 DROP 语句计数

系统存储过程sp_MSforeachtable和sp_MSforeachdb使用说明

数据库中表内容的删除总结(总有你需要的)

tomcat

SQL Server 如何清空数据库所有表数据

是否可以在 Matlab 中通过引用来解析结构?