并发访问数据库 - 阻止两个用户获取相同的值

Cha*_*lts 6 c# sql-server concurrency entity-framework-4

我有一个包含序号的表(想想发票号或学生证).

在某些时候,用户需要请求前一个号码(以便计算下一个号码).一旦用户知道当前号码,他们就需要生成下一个号码并将其添加到表格中.

我担心的是,由于并发访问,两个用户将能够错误地生成两个相同的号码.

我听说过存储过程,我知道这可能是一个解决方案.这里有最佳实践,以避免并发问题吗?

编辑:这是我到目前为止所拥有的:

USE [master]
GO

SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER PROCEDURE [dbo].[sp_GetNextOrderNumber]
AS
BEGIN
    BEGIN TRAN

DECLARE @recentYear INT
DECLARE @recentMonth INT
DECLARE @recentSequenceNum INT

-- SET NOCOUNT ON added to prevent extra result sets from
-- interfering with SELECT statements.
SET NOCOUNT ON;

    -- get the most recent numbers
SELECT @recentYear = Year, @recentMonth = Month, @recentSequenceNum = OrderSequenceNumber
FROM dbo.OrderNumbers 
WITH (XLOCK)
WHERE Id = (SELECT MAX(Id) FROM dbo.OrderNumbers)

    // increment the numbers
IF (YEAR(getDate()) > IsNull(@recentYear,0))
    BEGIN
        SET @recentYear = YEAR(getDate());
        SET @recentMonth = MONTH(getDate());
        SET @recentSequenceNum = 0;
    END
ELSE
    BEGIN
        IF (MONTH(getDate()) > IsNull(@recentMonth,0))
            BEGIN
                SET @recentMonth = MONTH(getDate());
                SET @recentSequenceNum = 0;
            END
        ELSE
            SET @recentSequenceNum = @recentSequenceNum + 1;
    END 

-- insert the new numbers as a new record   
INSERT INTO dbo.OrderNumbers(Year, Month, OrderSequenceNumber)
VALUES (@recentYear, @recentMonth, @recentSequenceNum)

COMMIT TRAN
END
Run Code Online (Sandbox Code Playgroud)

这似乎有效,并给了我想要的价值.到目前为止,我还没有添加任何锁定来防止并发访问.

编辑2:添加WITH(XLOCK)以锁定表,直到事务完成.我不打算在这里表演.只要我没有添加重复的条目,并且没有发生死锁,这应该可行.

Die*_*ego 6

你知道SQL Server会为你做这件事,对吧?如果需要序列号或计算列,如果需要根据另一个计算新值,则可以使用标识列.

但是,如果这不能解决您的问题,或者您需要进行复杂的计算以生成无法在简单插入中完成的新数字,我建议编写一个锁定表的存储过程,获取最后一个值,生成新的,插入然后解锁表.

阅读此链接以了解事务隔离级别

只要确保尽可能小的"锁定"时间