c# - 并发访问数据库——防止两个用户获取相同的值

标签 c# sql-server entity-framework-4 concurrency

我有一个带有序列号的表格(想想发票号或学生证)。

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

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

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

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

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

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

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

最佳答案

您知道 SQL Server 会为您做这些,对吗?如果您需要序列号,您可以使用标识列;如果您需要根据另一个值计算新值,则可以使用计算列。

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

阅读此 link了解事务隔离级别

只要确保“锁定”期尽可能短

关于c# - 并发访问数据库——防止两个用户获取相同的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9763925/

相关文章:

entity-framework - 在SaveChanges之前在AddObject之后查询对象?

c# - 为什么 EF 试图在 id 列中插入 NULL?

c# - 如何返回依赖于 using 语句链的对象?

c# - 填充 dataGridView 感谢 backGroundWorker

sql - 将 ID 列添加到查询结果中

mysql - 通过比较前后记录创建自定义 session 标识符

c# - 选择整个实体时, Entity Framework 返回过时的数据,但仅选择一个字段时, Entity Framework 返回最新数据

c# - 无法创建 list 资源名称

c# - 如何使用 UWP 在图像上绘制文本并保存(适用于 Windows 10)

php - 将 mySQL 数据库的日志文件日期时间格式从 12 小时制批量转换为 24 小时制