我有一个带有序列号的表格(想想发票号或学生证)。
在某些时候,用户需要请求上一个数字(以便计算下一个数字)。用户知道当前号码后,他们需要生成下一个号码并将其添加到表中。
我担心的是,由于并发访问,两个用户将能够错误地生成两个相同的数字。
我听说过存储过程,我知道这可能是一种解决方案。这里有避免并发问题的最佳实践吗?
编辑:这是我目前所拥有的:
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/