我正在使用动态 SQL 将默认值插入表中,并希望返回SCOPE_IDENTITY()
代理键的值。但是,由于某种原因,它SCOPE_IDENTITY()
返回了NULL
。我遗漏了什么吗?
谢谢!
DECLARE @SQL NVARCHAR(MAX),
@SurrogateKey INT,
@SchemaName VARCHAR(128) = 'dim',
@TableName VARCHAR(128) = 'Sales',
@NaturalKeyColumn VARCHAR(128) = 'SalesID',
@NaturalKeyValue VARCHAR(40) = '100',
@SurrogateKeyColumn VARCHAR(128) = 'SalesKey';
SET @SQL = N'SELECT @SurrogateKey = ' + QUOTENAME(@SurrogateKeyColumn) +
' FROM ' + QUOTENAME(@SchemaName) + '.' + QUOTENAME(@TableName) +
' WHERE ' + QUOTENAME(@NaturalKeyColumn) + ' = @NaturalKeyValue';
EXEC sp_executesql @SQL, N'@SurrogateKey INT OUTPUT, @NaturalKeyValue VARCHAR(40)', @SurrogateKey OUTPUT, @NaturalKeyValue;
IF @SurrogateKey IS NULL
BEGIN
BEGIN TRANSACTION;
SET @SQL = N'INSERT INTO ' + QUOTENAME(@SchemaName) + '.' + QUOTENAME(@TableName) +
' (' + QUOTENAME(@NaturalKeyColumn) + ', InferredFlag)
VALUES (@NaturalKeyValue, 1)';
EXEC sp_executesql @SQL, N'@NaturalKeyValue VARCHAR(40)', @NaturalKeyValue;
-- Check if the INSERT was successful
IF @@ROWCOUNT > 0
BEGIN
SET @SQL = N'SELECT @SurrogateKey = SCOPE_IDENTITY()';
EXEC sp_executesql @SQL, N'@SurrogateKey INT OUTPUT', @SurrogateKey OUTPUT;
END
COMMIT TRANSACTION;
END
SELECT @SurrogateKey;
SCOPE_IDENTITY()
不出所料,它只在单个范围内有效。您需要创建并执行单个动态 SQL 批处理。这也更加高效。您的代码的其他问题:
SELECT
。SELECT
,否则将采用共享锁,这对您没有帮助。您需要SERIALIZABLE
和UPDLOCK
这里。sysname
变量中,这是的别名nvarchar(128)
。XACT_ABORT ON
应使用以便在出现错误时正确进行回滚。SELECT
直接退出动态 SQL,您不需要OUT
参数,除非您之后在静态端需要该值。