我应该如何获得插入行的身份?

我知道@@IDENTITY和IDENT_CURRENT和SCOPE_IDENTITY,但不理解它们所附带的含义或影响。

谁能解释一下它们的区别,以及我什么时候使用它们?


从MSDN

@@IDENTITY、SCOPE_IDENTITY和IDENT_CURRENT是类似的函数,它们返回插入到表的IDENTITY列中的最后一个值。 @@IDENTITY和SCOPE_IDENTITY将返回当前会话中在任何表中生成的最后一个标识值。然而,SCOPE_IDENTITY只返回当前范围内的值;@@IDENTITY不受特定范围的限制。 IDENT_CURRENT不受作用域和会话的限制;它仅限于指定的表。IDENT_CURRENT返回在任何会话和任何范围内为特定表生成的标识值。有关更多信息,请参见IDENT_CURRENT。

IDENT_CURRENT是一个以表作为参数的函数。 当你在表上有一个触发器时,@@IDENTITY可能返回令人困惑的结果 大多数时候SCOPE_IDENTITY是你的英雄。


@@IDENTITY返回当前会话中为所有作用域中的任何表生成的最后一个标识值。这里需要小心,因为它是跨作用域的。您可以从触发器获得一个值,而不是当前语句。 SCOPE_IDENTITY()返回当前会话和当前作用域中为任何表生成的最后一个标识值。通常是你想用的。 IDENT_CURRENT('tableName')返回在任何会话和任何范围内为特定表生成的最后一个标识值。这允许您指定需要从哪个表中获取值,以防上面两个表不是您所需要的(非常罕见)。另外,正如@Guy Starbuck所提到的,“如果您想获得尚未插入记录的表的当前IDENTITY值,则可以使用此方法。” INSERT语句的OUTPUT子句将允许您访问通过该语句插入的每一行。因为它的作用域是特定的语句,所以它比上面的其他函数更直接。但是,它有点啰嗦(您需要将变量/temp表插入到表中,然后再查询),即使在回滚语句的错误场景中,它也会给出结果。也就是说,如果您的查询使用并行执行计划,这是获得标识的唯一保证方法(除了关闭并行)。但是,它在触发器之前执行,不能用于返回触发器生成的值。


@@IDENTITY是使用当前SQL连接插入的最后一个标识。这是从插入存储过程中返回的一个很好的值,在该存储过程中,您只需要为新记录插入标识,而不关心之后是否添加了更多行。

SCOPE_IDENTITY是使用当前SQL Connection插入的最后一个标识,并且在当前作用域中——也就是说,如果在插入之后根据触发器插入了第二个identity,那么它将不会反映在SCOPE_IDENTITY中,只反映在您执行的插入中。坦率地说,我从来没有理由使用它。

IDENT_CURRENT(tablename)是插入的最后一个标识,无论连接或作用域如何。如果您想获取未插入记录的表的当前IDENTITY值,则可以使用此方法。


我和其他人说的是一样的,所以每个人都是对的,我只是想让它更清楚。

@@IDENTITY返回客户端连接到数据库时插入的最后一个对象的id。 大多数情况下,这工作得很好,但有时触发器会插入一个您不知道的新行,您将从这个新行获得ID,而不是您想要的行

SCOPE_IDENTITY()解决了这个问题。它返回您在发送给数据库的SQL代码中插入的最后一个内容的id。如果触发器去创建额外的行,它们不会导致返回错误的值。万岁

IDENT_CURRENT返回任何人插入的最后一个ID。如果其他应用在不幸的时间插入了另一行,你会得到那一行的ID,而不是你的。

如果想要安全起见,请始终使用SCOPE_IDENTITY()。如果您坚持使用@@IDENTITY,然后有人决定稍后添加一个触发器,那么您的所有代码都将崩溃。


总是使用scope_identity(),永远不需要其他任何东西。


我认为检索插入的id最安全、最准确的方法是使用output子句。

例如(摘自以下MSDN文章)

使用AdventureWorks2008R2; 去 声明@MyTableVar表(NewScrapReasonID smallint, 名字varchar (50), ModifiedDate datetime); 插入生产。ScrapReason 输出插入。ScrapReasonID,插入。名字,插入。ModifiedDate 到@MyTableVar VALUES (N'操作符错误',GETDATE()); ——显示表变量的结果集。 SELECT NewScrapReasonID, Name, ModifiedDate FROM @ mytablear; ——显示表的结果集。 SELECT ScrapReasonID, Name, ModifiedDate 从Production.ScrapReason; 去


获取新插入行的标识的最好(也就是最安全的)方法是使用output子句:

create table TableWithIdentity
           ( IdentityColumnName int identity(1, 1) not null primary key,
             ... )

-- type of this table's column must match the type of the
-- identity column of the table you'll be inserting into
declare @IdentityOutput table ( ID int )

insert TableWithIdentity
     ( ... )
output inserted.IdentityColumnName into @IdentityOutput
values
     ( ... )

select @IdentityValue = (select ID from @IdentityOutput)

Add

SELECT CAST(scope_identity() AS int);

插入SQL语句的末尾

NewId = command.ExecuteScalar()

会取回它。


在你的插入语句之后,你需要添加这个。确认插入数据的表名。您将得到当前行,而不是刚才插入语句所影响的行。

IDENT_CURRENT('tableName')

我不能与其他版本的SQL Server对话,但在2012年,直接输出工作得很好。您不需要为临时表而烦恼。

INSERT INTO MyTable
OUTPUT INSERTED.ID
VALUES (...)

顺便说一下,这种技术也适用于插入多行。

INSERT INTO MyTable
OUTPUT INSERTED.ID
VALUES
    (...),
    (...),
    (...)

输出

ID
2
3
4

创建一个uuid并将其插入到列中。然后,您可以很容易地用uuid标识行。这是唯一可以实现的100%有效的解决方案。所有其他的解都太复杂了,或者在相同的边缘情况下不起作用。 例如:

1)创建行

INSERT INTO table (uuid, name, street, zip) 
        VALUES ('2f802845-447b-4caa-8783-2086a0a8d437', 'Peter', 'Mainstreet 7', '88888');

2)获取创建行

SELECT * FROM table WHERE uuid='2f802845-447b-4caa-8783-2086a0a8d437';

另一种保证所插入行的身份的方法是指定身份值,并使用SET IDENTITY_INSERT ON和OFF。这保证了您确切地知道标识值是什么!只要这些值没有被使用,就可以将这些值插入到标识列中。

CREATE TABLE #foo 
  ( 
     fooid   INT IDENTITY NOT NULL, 
     fooname VARCHAR(20) 
  ) 

SELECT @@Identity            AS [@@Identity], 
       Scope_identity()      AS [SCOPE_IDENTITY()], 
       Ident_current('#Foo') AS [IDENT_CURRENT] 

SET IDENTITY_INSERT #foo ON 

INSERT INTO #foo 
            (fooid, 
             fooname) 
VALUES      (1, 
             'one'), 
            (2, 
             'Two') 

SET IDENTITY_INSERT #foo OFF 

SELECT @@Identity            AS [@@Identity], 
       Scope_identity()      AS [SCOPE_IDENTITY()], 
       Ident_current('#Foo') AS [IDENT_CURRENT] 

INSERT INTO #foo 
            (fooname) 
VALUES      ('Three') 

SELECT @@Identity            AS [@@Identity], 
       Scope_identity()      AS [SCOPE_IDENTITY()], 
       Ident_current('#Foo') AS [IDENT_CURRENT] 

-- YOU CAN INSERT  
SET IDENTITY_INSERT #foo ON 

INSERT INTO #foo 
            (fooid, 
             fooname) 
VALUES      (10, 
             'Ten'), 
            (11, 
             'Eleven') 

SET IDENTITY_INSERT #foo OFF 

SELECT @@Identity            AS [@@Identity], 
       Scope_identity()      AS [SCOPE_IDENTITY()], 
       Ident_current('#Foo') AS [IDENT_CURRENT] 

SELECT * 
FROM   #foo 

如果您正在从另一个数据源加载数据或合并来自两个数据库的数据等,这可能是一个非常有用的技术。


尽管这是一个较旧的线程,但有一种较新的方法可以做到这一点,它可以避免旧版本SQL Server中IDENTITY列的一些缺陷,比如服务器重新启动后标识值的空白。序列在SQL Server 2016和转发中可用,这是一种较新的方法,使用TSQL创建SEQUENCE对象。这允许您在SQL Server中创建自己的数字序列对象,并控制它如何递增。

这里有一个例子:

CREATE SEQUENCE CountBy1  
    START WITH 1  
    INCREMENT BY 1 ;  
GO  

然后在TSQL中,您可以执行以下操作来获得下一个序列ID:

SELECT NEXT VALUE FOR CountBy1 AS SequenceID
GO

下面是CREATE SEQUENCE和NEXT VALUE FOR的链接