锁不住的查询

最近在处理一个锁的问题时,发现一个比较郁闷的事,使用X锁居然无法锁住查询,模拟这个问题,可以使用如下T-SQL脚本来建立测试环境。

USE master;GOIF @@TRANCOUNT > 0ROLLBACK TRAN;GO-- =======================================-- 建立测试数据库-- a. 删除测试库, 如果已经存在的话IF DB_ID(N'db_xlock_test') IS NOT NULLBEGIN;ALTER DATABASE db_xlock_testSET SINGLE_USERWITHROLLBACK AFTER 0;DROP DATABASE db_xlock_test;END;-- b. 建立测试数据库CREATE DATABASE db_xlock_test;-- c. 关闭READ_COMMITTED_SNAPSHOT 以保持SELECT 的默认加锁模式ALTER DATABASE db_xlock_testSET READ_COMMITTED_SNAPSHOT OFF;GO-- =======================================-- 建立测试表USE db_xlock_test;GOCREATE TABLE dbo.tb(id int IDENTITYPRIMARY KEY,name sysname);INSERT dbo.tbSELECT TOP(50000)O1.name + N'.' + O2.name + N'.' + O3.nameFROM sys.objects O1 WITH(NOLOCK),sys.objects O2 WITH(NOLOCK),sys.objects O3 WITH(NOLOCK);GO

然后,建立一个连接,执行下面的脚本来实现加锁。

-- =======================================-- 测试连接1 - 加锁BEGIN TRAN--测试的初衷是通过SELECT加锁,结果发现UPDATE也锁不住UPDATE dbo.tb SET name = name--SELECT COUNT(*) FROM dbo.tb WITH(XLOCK)WHERE id <= 2;SELECTspid = @@SPID,tran_count = @@TRANCOUNT,database_name = DB_NAME(),object_id = OBJECT_ID(N'dbo.tb', N'Table');-- 显示锁EXEC sp_lock@@SPID;

通过执行结果,可以看到对象被加锁的情况:表级和页级上是IX锁,记录上是X锁。

spid

tran_count

database_name

object_id

51

1

db_xlock_test

21575115

spid

dbid

ObjId

IndId

Type

Resource

Mode

Status

51

7

0

0

DB

S

GRANT

51

7

21575115

1

PAG

0.095138889

IX

GRANT

51

7

21575115

0

TAB

IX

GRANT

51

1

1131151075

0

TAB

IS

GRANT

51

7

21575115

1

KEY

(020068e8b274)

X

GRANT

51

7

21575115

1

KEY

-10086470766

X

GRANT

           

然后新建一个连接,执行下面的T-SQL查询,看看会否被连接1锁住

-- =======================================-- 测试连接2 - 被阻塞(在测试连接1 执行后执行)SET TRANSACTION ISOLATION LEVEL READ COMMITTED;SELECT * FROM dbo.tbWHERE id <= 2;

上述查询会很快返回结果,并不会被查询1阻塞住。

按照我们的了解(联机帮助上也有说明),在READ COMMITTED事务隔离级别下,查询使用共享锁(S),而根据锁的兼容级别,S锁是与X锁冲突的,所以正常情况下,连接2的查询需要等待连接1执行完成。可是测试的结果去违反了这一原则。

为了了解为什么连接2不会被阻塞,对连接2做了一个Trace,发现一个更郁闷的问题,Trace的结果如下:

EventClass

TextData

ObjectID

Type

Mode

Lock:Acquired

 

21575115

5 – OBJECT

6 – IS

Lock:Acquired

1:77

0

6 – PAGE

6 – IS

Lock:Acquired

[PLANGUIDE]

0

2 – DATABASE

3 – S

Lock:Acquired

 

21575115

5 – OBJECT

6 – IS

Lock:Acquired

1:77

0

6 – PAGE

6 – IS

Lock:Acquired

1:80

0

6 – PAGE

6 – IS

Lock:Acquired

1:89

0

6 – PAGE

6 – IS

Trace的前面两行是连接2的Trace结果,从结果看,连接2仅使用了意向共享锁(IS),而且只是表级和页级,按照锁的兼容性原则,IS和IX(连接1在表级和页级仅使用了IX锁)是不冲突的,所以连接2的查询不会被阻塞。在增加了查询的数据量后,Trace结果表明查还是只在表级和页级使用了IS锁(Trace结果的最后4行)。

对于这个问题,解决的办法当然就是提升连接1锁的粒度,使用PAGLOCK表提示将锁的粒度提升到页级,这样IS与X是冲突的,就可以成功阻塞连接2。

但疑问就是,为什么查询只在表级和页级下意向共享锁(IS),而不在行级下共享锁(X),这个似乎与联机帮助上的说明不一样(还是一直以来理解上的偏差呢)。

附:联机帮助上关于锁模式的说明

共享锁

共享锁(S 锁)允许并发事务在封闭式并发控制下读取 (SELECT) 资源。

更新锁

更新锁(U 锁)可以防止常见的死锁。在可重复读或可序列化事务中,此事务读取数据 [获取资源(页或行)的共享锁(S 锁)],然后修改数据 [此操作要求锁转换为排他锁(X 锁)]。如果两个事务获得了资源上的共享模式锁,然后试图同时更新数据,则一个事务尝试将锁转换为排他锁(X 锁)。共享模式到排他锁的转换必须等待一段时间,因为一个事务的排他锁与其他事务的共享模式锁不兼容;发生锁等待。第二个事务试图获取排他锁(X 锁)以进行更新。由于两个事务都要转换为排他锁(X 锁),并且每个事务都等待另一个事务释放共享模式锁,因此发生死锁。

若要避免这种潜在的死锁问题,请使用更新锁(U 锁)。一次只有一个事务可以获得资源的更新锁(U 锁)。如果事务修改资源,则更新锁(U 锁)转换为排他锁(X 锁)。

排他锁

排他锁(X 锁)可以防止并发事务对资源进行访问。使用排他锁(X 锁)时,任何其他事务都无法修改数据;仅在使用 NOLOCK 提示或未提交读隔离级别时才会进行读取操作。

数据修改语句(如 INSERT、UPDATE 和 DELETE)合并了修改和读取操作。语句在执行所需的修改操作之前首先执行读取操作以获取数据。因此,数据修改语句通常请求共享锁和排他锁。例如,UPDATE 语句可能根据与一个表的联接修改另一个表中的行。在此情况下,除了请求更新行上的排他锁之外,UPDATE 语句还将请求在联接表中读取的行上的共享锁。

意向锁

数据库引擎使用意向锁来保护共享锁(S 锁)或排他锁(X 锁)放置在锁层次结构的底层资源上。意向锁之所以命名为意向锁,是因为在较低级别锁前可获取它们,因此会通知意向将锁放置在较低级别上。

本文讲解了锁不住的查询,更多相关内容,请关注创想鸟。

相关推荐:

讲解更新锁(U)与排它锁(X)的相关知识

SQL Server 2008 处理隐式数据类型转换在执行计划中的增强

如何让MySQL中单句实现无限层次父子关系查询

以上就是锁不住的查询的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/383579.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月6日 12:31:32
下一篇 2025年11月6日 12:32:22

相关推荐

  • 阐述什么是CSS3?

    网页制作Webjx文章简介:CSS3不是新事物,更不是只是围绕border-radius属性实现的圆角。它正耐心的坐在那里,已经准备好了首次登场,呷着咖啡,等着浏览器来铺上红地毯。            CSS3不是新事物,更不是只是围绕border-radius属性实现              …

    好文分享 2025年12月23日
    000
  • 如何在Selenium测试中设置HTML元素的样式显示?

    我们可以使用 Selenium webdriver 设置 html 元素的样式显示。 DOM 在 Javascript 的帮助下与页面上的元素进行交互。 Selenium 通过 executeScript 方法执行 Javascript 命令。要执行的命令作为参数传递给该方法。 一些操作(例如设置样…

    2025年12月21日
    000
  • 如何将 C++ 框架与自动化测试集成?

    将 c++++ 框架与自动化测试集成可增强代码覆盖率、提供快速反馈并节省时间和精力。步骤包括:选择测试框架、使用框架 api、集成到框架、运行测试。 如何将 C++ 框架与自动化测试集成 引言C++ 框架提供了一个可扩展、可重用的组件集合,简化了应用程序的开发。将自动化测试与 C++ 框架集成至关重…

    2025年12月18日
    000
  • 如何在C++应用程序中使用框架进行测试?

    在 c++++ 应用程序中使用框架进行测试可以提高测试的可重复性、简化维护并提供跨平台兼容性。步骤包括:选择框架,集成框架,编写测试,执行测试和评估结果。使用 google test 等框架可以简化测试,例如测试计算给定数组和的函数时,可以使用 expect_eq 断言来验证计算的和是否等于预期的结…

    2025年12月18日
    000
  • 如何测试和验证修改后的C++框架的正确性?

    为了测试和验证修改后的 c++++ 框架的正确性,需要执行以下步骤:单元测试:为单个组件编写测试用例。集成测试:测试组件之间的协作。冒烟测试:验证基本功能。端到端测试:模拟应用程序使用。 如何测试和验证修改后的 C++ 框架的正确性? 在修改了现有的 C++ 框架后,至关重要的是测试和验证其正确性,…

    2025年12月18日
    000
  • 使用 C++ 框架简化测试和调试过程

    使用 gtest 和 gmoc++k c++ 框架可以简化测试和调试:gtest:清晰且简洁的单元测试框架,用于编写可读的测试用例。gmock:用于创建模拟和存根对象,以测试依赖于外部系统的代码。实战案例:gtest 和 gmock 可用于测试类操作,例如加法和减法,并可使用 gmock 模拟依赖关…

    2025年12月18日
    000
  • C++ 框架测试实践:自动化测试策略的实施指南

    C++ 框架测试实践:自动化测试策略的实施指南 引言 在现代软件开发中,自动化测试对于确保代码的健壮性和可靠性至关重要。本文将探讨适用于 C++ 框架的自动化测试策略,提供一步一步的指南,并附上实战案例。 选择自动化测试工具 立即学习“C++免费学习笔记(深入)”; 第一步是选择一个自动化测试框架。…

    2025年12月18日
    000
  • C++ 框架中可重用组件的测试和验证策略

    为了确保 c++++ 框架中可重用组件的质量,本文建议采用以下测试和验证策略:单元测试:对单个组件进行隔离测试以识别内部缺陷。集成测试:测试多个组件的协同工作,发现接口不匹配或依赖关系问题。性能测试:评估组件在不同负载下的行为,识别资源瓶颈和提高性能的机会。 C++ 框架中可重用组件的测试和验证策略…

    2025年12月18日
    000
  • 函数指针在 C++ 中如何提升代码的可测试性?

    函数指针增强了代码的可测试性,通过以下方式:隔离代码,使测试易于维护。降低耦合度,提高代码灵活性。函数指针使将函数作为变量传递和存储成为可能,通过解引用函数指针即可调用函数。这将业务逻辑与测试用例分离,提高了测试用例的灵活性,降低了代码耦合度。 函数指针:增强 C++ 代码可测试性的利器 理解函数指…

    2025年12月18日
    000
  • 闭包在测试和调试方面的作用是什麼?

    闭包在测试和调试中的作用包括:隔离测试,防止外部变量影响结果。调试难以到达的变量,保持对变量的访问和修改。缓存数据,提升程序性能。 闭包在测试和调试中的作用 什么是闭包? 闭包是一个函数,它能访问它定义所在作用域之外的变量。闭包将这些外部变量保存在内存中,即使定义它们的函数已执行完毕。 闭包在测试和…

    2025年12月18日
    000
  • 如何进行C++代码的性能测试?

    如何进行C++代码的性能测试? 概述:在软件开发过程中,性能测试是一项非常重要的任务。对于C++代码来说,性能测试可以帮助开发人员了解代码的执行效率,找到性能瓶颈,并对其进行优化。本文将介绍一些常用的C++代码性能测试方法和工具,帮助开发人员提高代码性能。 测试方法:1.时间测量: C++代码性能测…

    2025年12月17日
    000
  • .net中关于异步性能测试的示例代码

    很久没有写博客了,今年做的产品公司这两天刚刚开了发布会,稍微清闲下来,想想我们做的产品还有没有性能优化空间,于是想到了.net的异步可以优化性能,但到底能够提升多大的比例呢?恰好有一个朋友正在做各种语言的异步性能测试(有关异步和同步的问题,请参考客《aio与bio接口性能对比》),于是我今天写了一个…

    2025年12月17日 好文分享
    000
  • Golang测试表驱动与基准组合方法

    表驱动测试结合基准测试可同时验证代码正确性与性能。通过定义测试用例结构体,TestAdd函数覆盖多种输入场景,确保逻辑正确;BenchmarkConcatStrings则对不同规模字符串拼接进行性能测量,利用b.Run为每组数据单独计时,实现精细化性能监控,提升测试可维护性与执行效率。 在 Go 语…

    2025年12月16日
    000
  • Golang如何实现测试数据初始化与清理

    使用TestMain可全局初始化与清理测试资源,如数据库连接和测试数据;每个测试函数可通过defer实现独立的初始化与清理;通过结构体封装Setup/Teardown方法可模拟测试套件,共享资源并控制生命周期;建议用事务回滚避免数据污染,确保清理逻辑幂等且不因panic失效。 在Go语言中,测试数据…

    2025年12月16日
    000
  • Golang如何测试并发安全函数

    答案是使用go test -race结合sync.WaitGroup模拟多协程并发访问,验证数据一致性和竞态条件。通过启动多个goroutine对共享资源进行操作,利用WaitGroup同步等待所有操作完成,并借助-race检测是否存在内存竞争,若存在则测试失败。示例中对SafeCounter的In…

    2025年12月16日
    000
  • 如何使用Golang编写集成测试

    集成测试重点是验证多组件协同行为,需使用真实依赖如数据库和HTTP服务。1. 区分单元与集成测试,文件命名用*_integration_test.go;2. 用//go:build integration标签控制执行;3. TestMain中启动服务并等待,注意端口配置;4. setup/teard…

    2025年12月16日
    000
  • Golang单元测试与集成测试结合方法

    单元测试与集成测试结合是Go项目质量保障的核心策略。单元测试通过表格驱动和Mock接口快速验证函数逻辑,确保代码内部正确性;集成测试利用Docker Compose或testcontainers-go启动真实依赖,通过TestMain管理环境生命周期,验证组件协作。两者分层互补,共同构建高效可靠的测…

    好文分享 2025年12月15日
    000
  • Golang测试环境搭建 编写测试用例指南

    Go语言的测试体验体现了其简洁高效的设计哲学,核心是使用内置的testing包,无需额外框架即可编写单元、基准和示例测试。通过遵循_test.go文件命名规范和TestXxx函数格式,结合go test命令运行测试。推荐采用表驱动测试和子测试(t.Run)提升可维护性,利用接口模拟外部依赖以实现隔离…

    2025年12月15日
    000
  • Golang测试文件命名规范 _test.go文件作用解析

    Go语言中测试文件必须以_test.go结尾,这是go test命令自动发现和执行测试的强制约定。该命名方式实现了工具链的无缝集成,使测试文件在项目中具有高可读性,并确保测试代码与生产代码隔离,避免污染最终构建产物。测试函数需遵循特定命名规则:以TestXxx格式命名的单元测试函数使用testing…

    2025年12月15日
    000
  • Python开发经验总结:提高代码复用和可维护性的技巧

    Python作为一种广泛应用的编程语言,被广泛应用于Web开发、数据分析、人工智能等领域。在Python开发工作中,不仅要求高效完成项目任务,还要注重代码的可维护性和复用性。本文将总结提高Python代码复用和可维护性的一些技巧和经验,并希望能对Python开发者有所帮助。 1. 使用函数和模块 在…

    2025年12月13日
    000

发表回复

登录后才能评论
关注微信