2

这不是一个完整的瓶子,但我的印象是@@IDENTITY 给出了在数据库中创建的最后一个身份的值,无论在哪里。SCOPE_IDENTITY() 给出当前语句范围内的值。

我有一个带有标识列的表。没有触发器,只有一个普通表和主索引。我已在两个单独的浏览器选项卡中运行此 Web 应用程序代码:

Dim connDb As New SqlConnection(myconnectionstring)
Dim transDb As SqlTransaction = connDb.BeginTransaction()
Dim cmdInsert As New SqlCommand("INSERT INTO mytable (somecolumn) VALUES (somevalue)", conn, transDb)
Dim cmdGetIdentity As New SqlCommand("SELECT @@IDENTITY", connDb, transDb)
' Add the record, pause, get identity.
Dim iNewHosp As Int32 = cmdInsert.ExecuteNonQuery() ' Returns 1 correctly.
Threading.Thread.Sleep(5000) ' Give the other tab time to add a record.
Dim iNewID As Int32 = cmdGetIdentity.ExecuteScalar ' Returns the new ID.
' Commit trans, close, dispose, etc..

请注意,这都是在事务中完成的,不确定这是否会有所不同。在这里使用 SQL Server 2000(不幸的是)。在 SQL2008 中也尝试过,同样的事情发生了。

因此,我在一个浏览器选项卡中运行该页面,然后在它处于睡眠状态时,在另一个选项卡中再次运行它,这会添加另一条记录。我看到的“问题”是第一个选项卡随后返回其 INSERT的正确ID,另一个选项卡也是如此。我认为第一个选项卡会返回最近插入的 ID,即。在第二个选项卡中完成的那个?

我最初尝试为此使用 SCOPE_IDENTITY(),但它返回 NULL,因此我尝试@@IDENTITY 只是为了看看发生了什么。所以我有 2 个问题 - 为什么 SCOPE_ 返回 NULL,为什么 @@IDENTITY 返回特定于范围的结果?

4

2 回答 2

4

不,@@IDENTITY仍然仅限于在同一会话中生成的身份值。请参阅中的讨论SCOPE_IDENTITY

SCOPE_IDENTITY并返回当前会话@@IDENTITY中任何表中生成的最后一个标识值。但是,返回仅在当前范围内插入的值;不限于特定范围。SCOPE_IDENTITY@@IDENTITY

(重点补充)

范围≠会话。在这里,范围指的是例如将返回由触发器内的一个@@IDENTITY生成的标识值。INSERT您还应该看到,您的最终选择是IDENT_CURRENT让您查询特定表中的最新标识值,而不管会话如何。

于 2012-05-17T08:32:24.010 回答
0

您正在 Transacation 中插入记录,因此SCOPE_IDENTITY仅限于该交易。因此,您无法看到从其他交易中插入的价值。

当为第二个选项卡提交事务时,您将能够看到第二个选项卡插入的标识。

SELECT SCOPE_IDENTITY()

它返回连接上和同一范围内的语句生成的最后一个 IDENTITY 值,而不管生成该值的表是什么。SCOPE_IDENTITY() 与@@IDENTITY 一样,将返回在当前会话中创建的最后一个身份值,但它也会将其限制在您当前的范围内。换句话说,它将返回您显式创建的最后一个身份值,而不是由触发器或用户定义的函数创建的任何身份。

于 2012-05-17T08:35:31.917 回答