AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / dba / 问题 / 334280
Accepted
The amateur programmer
The amateur programmer
Asked: 2023-12-22 19:03:07 +0800 CST2023-12-22 19:03:07 +0800 CST 2023-12-22 19:03:07 +0800 CST

了解 MS SQL TSQL 事务中的表锁定

  • 772

我想让旧的后续查询在并行运行时免受数据竞争的影响。该查询使用特定条件检查表中是否存在一行,如果不存在该行,则会插入包含新数据的新行。旧查询粘贴如下:

BEGIN
    DECLARE @txtPer VARCHAR(MAX) = @nro;
    DECLARE @txtCmin VARCHAR(MAX) = @min;
    DECLARE @txtCmax VARCHAR(MAX) = @max;
    IF NOT EXISTS (SELECT 1 FROM SEND WHERE MSG LIKE (@txtPer + '%') AND STATE < 2 AND ID = 1)
    BEGIN
        ----Time of check is not time of use
        ----Someone could possibly do another insert before this == data race
        INSERT INTO SEND (SNDID, ID, MSGCODE, MSG, STATE, INFO, INFO_TEXT, CHANGEDATE, CREATEDATE)
        SELECT MAX(SNDID)+10,1,1,(@txtPer + @txtCmin + '           ' + @txtCmax),0,0,' ',getdate(),getdate() FROM SEND
    END
END

我想出了一个稍微干净一点的新版本,并使用独占表锁:

BEGIN TRANSACTION;
DECLARE @txtPer VARCHAR(MAX) = @nro;
DECLARE @txtCmin VARCHAR(MAX) = @min;
DECLARE @txtCmax VARCHAR(MAX) = @max;

DECLARE @IdMax INT
--Get MAX and simultaneously acquire lock for the table to prevent modifications during this transaction?
SELECT @IdMax=MAX(SNDID) FROM SEND WITH(TABLOCKX)

IF NOT EXISTS(SELECT 1 FROM SEND WHERE MSG LIKE (@txtPer + '%') AND STATE < 2 AND ID = 1)
BEGIN
    INSERT INTO SEND (SNDID, ID, MSGCODE, MSG, STATE, INFO, INFO_TEXT, CHANGEDATE, CREATEDATE)
    VALUES((@IdMax + 10),1,1,(@txtPer + @txtCmin + '           ' + @txtCmax),0,0,' ',getdate(),getdate())
END
COMMIT; --writes potential change and releases table lock?

我的理解是否正确,即在操作期间获取的表锁SELECT MAX()(可能还有同一事务中的其他表锁)一直保留到整个事务完成该COMMIT语句?DB 是旧的 MS SQL Server 2005。

t-sql
  • 2 2 个回答
  • 73 Views

2 个回答

  • Voted
  1. Best Answer
    Charlieface
    2023-12-25T01:28:30+08:002023-12-25T01:28:30+08:00

    我的理解是否正确:在 SELECT MAX() 操作(可能还有同一事务中的其他操作)期间获取的表锁一直保持到整个事务使用 COMMIT 语句完成为止?

    是的,只要它是TABLOCKX并且不是TABLOCK。仅当您使用悲观锁定而不是乐观快照锁定时,这才有效,如 @DavidBrowne 所提到的。

    然而,这是极其低效的。

    • 它将完全锁定整个表的读取或写入。而是使用UPDLOCKand SERIALIZABLE(或HOLDLOCK) 提示。
      • SERIALIZABLE即使数据不存在(即范围锁),提示也是必要的,以确保其他人无法插入。
      • UPDLOCK为了在表上放置 U 锁,该提示是必要的,这可以防止其他任何请求 U 锁的人继续进行,同时仍然允许只读查询。这往往可以防止可序列化隔离级别上的死锁。
    • ID, MSG, STATE确保您的表在列或 上建立索引ID, STATE, MSG,以便NOT EXISTS提高效率。
    • 它还会在整个表中搜索最大值SNDID,您应该使用IDENTITY列。
    • 用于XACT_ABORT ON确保在发生错误时正确回滚。
    SET XACT_ABORT ON;
    BEGIN TRAN;
    
    IF NOT EXISTS (SELECT 1
        FROM SEND WITH (UPDLOCK, SERIALIZABLE)
        WHERE MSG LIKE CONCAT(@nro, '%')
          AND STATE < 2
          AND ID = 1
    )
    BEGIN
            INSERT INTO SEND (ID, MSGCODE, MSG, STATE, INFO, INFO_TEXT, CHANGEDATE, CREATEDATE)
            VALUES (1, 1, CONCAT(@nro, @min, '           ', @max), 0, 0, ' ', getdate(), getdate());
    END;
    
    COMMIT;
    
    • 2
  2. David Browne - Microsoft
    2023-12-22T23:34:47+08:002023-12-22T23:34:47+08:00

    我的理解是否正确:在 SELECT MAX() 操作(可能还有同一事务中的其他操作)期间获取的表锁一直保持到整个事务使用 COMMIT 语句完成为止?

    是的。但是,如果数据库配置了 READ COMMITTED SNAPSHOT,则该锁不会阻止另一个会话读取该表。

    序列化这些的正确方法是在选择上使用锁定提示来决定该行是否存在,如下所示:

    SELECT 1 FROM SEND WITH(UPDLOCK,HOLDLOCK)...

    • 1

相关问题

  • 如何使用 TSQL 更改 SQL 服务器配置管理器设置?

  • 如何从结果集中获取列名和类型的列表?

  • MS SQL:使用计算值计算其他值

  • 如何判断 SQL Server 数据库是否仍在使用?

  • 实施 PIVOT 查询

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    连接到 PostgreSQL 服务器:致命:主机没有 pg_hba.conf 条目

    • 12 个回答
  • Marko Smith

    如何让sqlplus的输出出现在一行中?

    • 3 个回答
  • Marko Smith

    选择具有最大日期或最晚日期的日期

    • 3 个回答
  • Marko Smith

    如何列出 PostgreSQL 中的所有模式?

    • 4 个回答
  • Marko Smith

    列出指定表的所有列

    • 5 个回答
  • Marko Smith

    如何在不修改我自己的 tnsnames.ora 的情况下使用 sqlplus 连接到位于另一台主机上的 Oracle 数据库

    • 4 个回答
  • Marko Smith

    你如何mysqldump特定的表?

    • 4 个回答
  • Marko Smith

    使用 psql 列出数据库权限

    • 10 个回答
  • Marko Smith

    如何从 PostgreSQL 中的选择查询中将值插入表中?

    • 4 个回答
  • Marko Smith

    如何使用 psql 列出所有数据库和表?

    • 7 个回答
  • Martin Hope
    Jin 连接到 PostgreSQL 服务器:致命:主机没有 pg_hba.conf 条目 2014-12-02 02:54:58 +0800 CST
  • Martin Hope
    Stéphane 如何列出 PostgreSQL 中的所有模式? 2013-04-16 11:19:16 +0800 CST
  • Martin Hope
    Mike Walsh 为什么事务日志不断增长或空间不足? 2012-12-05 18:11:22 +0800 CST
  • Martin Hope
    Stephane Rolland 列出指定表的所有列 2012-08-14 04:44:44 +0800 CST
  • Martin Hope
    haxney MySQL 能否合理地对数十亿行执行查询? 2012-07-03 11:36:13 +0800 CST
  • Martin Hope
    qazwsx 如何监控大型 .sql 文件的导入进度? 2012-05-03 08:54:41 +0800 CST
  • Martin Hope
    markdorison 你如何mysqldump特定的表? 2011-12-17 12:39:37 +0800 CST
  • Martin Hope
    Jonas 如何使用 psql 对 SQL 查询进行计时? 2011-06-04 02:22:54 +0800 CST
  • Martin Hope
    Jonas 如何从 PostgreSQL 中的选择查询中将值插入表中? 2011-05-28 00:33:05 +0800 CST
  • Martin Hope
    Jonas 如何使用 psql 列出所有数据库和表? 2011-02-18 00:45:49 +0800 CST

热门标签

sql-server mysql postgresql sql-server-2014 sql-server-2016 oracle sql-server-2008 database-design query-performance sql-server-2017

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve