我最近发现了一个问题,我想使用 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
附言。复制此行为的代码如下:
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