我有一个INSTEAD OF UPDATE
触发器,它可以动态生成代码,然后执行它。问题是执行代码时,我总是收到错误消息:
无效的对象名称“已插入”
触发代码如下:
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER Trigger [dbo].[TACMasterLayouts_VersionVisibilityHandling_updates]
ON [dbo].[TACMasterLayouts_VersionVisibilityHandling]
INSTEAD OF UPDATE
AS
IF @@rowcount = 0
RETURN;
SET NOCOUNT ON
DECLARE @tableToUpdate AS NVARCHAR(MAX) = 'dbo.tacmasterlayouts';
DECLARE @UpdateStatement AS NVARCHAR(MAX)
DECLARE @IdentityField AS NVARCHAR(MAX) = (SELECT Name
FROM sys.columns
WHERE [object_id] = OBJECT_ID('dbo.tacmasterlayouts')
AND is_identity = 1)
SET @UpdateStatement = 'update ' + @tableToUpdate + ' set ';
DECLARE @Fields AS NVARCHAR(MAX)
SELECT @Fields = COALESCE(@Fields + ', ', '') + @tableToUpdate + '.' + Name + ' = inserted.' + Name
FROM sys.columns
WHERE [object_id] = OBJECT_ID('dbo.tacmasterlayouts')
AND Name != @IdentityField;
SET @UpdateStatement = @UpdateStatement + @fields
SET @UpdateStatement = @UpdateStatement + ' FROM ' + @tableToUpdate + ', inserted'
SET @UpdateStatement = @UpdateStatement + ' where ' + @tableToUpdate + '.' + @IdentityField + ' = inserted.' + @IdentityField
EXECUTE sp_executesql @UpdateStatement
动态代码如下:
UPDATE TACMasterLayouts SET
TACMasterLayouts.TACMasterLayoutName = inserted.TACMasterLayoutName,
TACMasterLayouts.TACMasterLayoutDestinationFileName = inserted.TACMasterLayoutDestinationFileName,
TACMasterLayouts.LayoutTypeCategoryId = inserted.LayoutTypeCategoryId,
TACMasterLayouts.LayoutTypeId = inserted.LayoutTypeId,
TACMasterLayouts.TACMasterLayoutDescription = inserted.TACMasterLayoutDescription,
TACMasterLayouts.TACMasterLayoutJasper = inserted.TACMasterLayoutJasper,
TACMasterLayouts.TACMasterLayoutJrxml = inserted.TACMasterLayoutJrxml,
TACMasterLayouts.TACMasterLayoutHtml = inserted.TACMasterLayoutHtml,
TACMasterLayouts.TACMasterLayoutDontClone = inserted.TACMasterLayoutDontClone,
TACMasterLayouts.TACMasterLayoutIsSubReport = inserted.TACMasterLayoutIsSubReport,
TACMasterLayouts.TACMasterLayoutVersion = inserted.TACMasterLayoutVersion,
TACMasterLayouts.TACMasterLayoutKey = inserted.TACMasterLayoutKey,
TACMasterLayouts.TACMasterLayoutTimestampLastModified = inserted.TACMasterLayoutTimestampLastModified,
TACMasterLayouts.TACMasterLayoutHash = inserted.TACMasterLayoutHash,
TACMasterLayouts.TACMasterLayoutTimestampLastCheckout = inserted.TACMasterLayoutTimestampLastCheckout,
TACMasterLayouts.TACMasterLayoutCheckedOutByUserId = inserted.TACMasterLayoutCheckedOutByUserId,
TACMasterLayouts.TACMasterLayoutIsCheckedIn = inserted.TACMasterLayoutIsCheckedIn,
TACMasterLayouts.TACMasterLayoutCheckedInByUserId = inserted.TACMasterLayoutCheckedInByUserId,
TACMasterLayouts.TACMasterLayoutCheckedOutFolderName = inserted.TACMasterLayoutCheckedOutFolderName
FROM TACMasterLayouts, inserted
WHERE TACMAsterlayouts.TACMasterLayoutId = inserted.TACMasterLayoutId;
提示:
如果我在触发器中静态输入动态生成的代码,则可以正常工作。所以代码本身还可以。
问题:为什么在这种情况下inserted
不可用?以及如何解决?
答案 0 :(得分:2)
执行动态sql时,它具有自己的作用域,如调用另一个存储过程,因此它不能引用触发器的作用域中的内容(如插入,删除或局部变量等)。您必须在动态sql字符串作为必需的键值,并且还准备处理插入的多于一行的行,这会触发触发器。
答案 1 :(得分:1)
从我发现的内容来看,您似乎无权以这种方式在动态SQL中使用inserted
表,但是您应该能够完成您要尝试的操作使用SQLCLR
答案 2 :(得分:0)
由于上述答案没有给我想要的解决方案,因此我进一步进行了挖掘,发现了以下解决方案(很酷)(感谢以下文章:https://www.sqlteam.com/forums/topic.asp?TOPIC_ID=11318):
如果需要将插入的表传递到动态SQL代码中(传递给sp_executesql存储过程),请执行以下操作:
select * from inserted into #inserted
,然后可以在#inserted下的sp_executesql中立即访问它。无需将此作为参数传递给sp_executesql。
最终的有效代码如下:
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER Trigger [dbo].[TACMasterLayouts_VersionVisibilityHandling_updates] on [dbo].[TACMasterLayouts_VersionVisibilityHandling] INSTEAD OF update
as
if @@rowcount = 0 return;
SET NOCOUNT ON
Declare @tableToUpdate AS Nvarchar(max) = 'dbo.tacmasterlayouts';
Declare @UpdateStatement AS Nvarchar(max)
Declare @IdentityField as Nvarchar(max) = (SELECT Name FROM sys.columns WHERE [object_id] = OBJECT_ID(@tableToUpdate) and is_identity = 1)
set @UpdateStatement = 'update ' + @tableToUpdate + ' set ';
Declare @Fields AS Nvarchar(MAX)
SELECT @Fields = COALESCE(@Fields + ', ', '') + @tableToUpdate + '.' + Name + ' = #inserted.' + Name FROM sys.columns WHERE [object_id] = OBJECT_ID(@tableToUpdate) and Name != @IdentityField;
set @UpdateStatement = @UpdateStatement + @fields
set @UpdateStatement = @UpdateStatement + ' FROM ' + @tableToUpdate + ', #inserted'
set @UpdateStatement = @UpdateStatement + ' where ' + @tableToUpdate + '.' + @IdentityField + ' = #inserted.' + @IdentityField
select * into #inserted from inserted
EXECUTE sp_executesql @UpdateStatement