SQL SERVER 锁升级的 investigation 与 别吃黄连

2024-03-05 23:20

本文主要是介绍SQL SERVER 锁升级的 investigation 与 别吃黄连,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前些日子在分析SQL SERVER 死锁的过程中,检查有一些莫名其妙的死锁,两个根本不搭噶的事务,锁在了一起,WHY,其实SQL SERVER 在数据库界,算是一朵奇葩,独有的锁升级的技术,我想你应该不曾听到 MYSQL ,ORACLE , PG  ,MONGODB 这些数据库提及到的锁升级的问题。

而牵扯到锁升级到额问题,就需要提及,到底为什么锁升级,锁从哪里升级到哪里的问题。所以就有了这篇文字,关于锁升级的东西。

首先我们在谈论锁的时候,的先站在一个讨论的起跑线,就是锁是发生在内存级别的,并且锁的开始和结束都是伴随着“事务”的开始和结束。在达成如上的共识后我们就开始下面的一些讨论和研究。

当事务管理器接收到提交请求时,它向事务中涉及的所有资源管理器发送一个prepare命令。然后,每个资源管理器执行使事务持久所需的所有操作,并将保存事务日志映像的所有缓冲区刷新到磁盘。当每个资源管理器完成准备阶段时,它将准备的成功或失败返回给事务管理器。

如果事务管理器从所有资源管理器接收到成功的准备,它将向每个资源管理器发送提交命令。然后,资源管理器可以完成提交。如果所有资源管理器都报告提交成功,那么事务管理器将向应用程序发送成功通知。如果任何资源管理器报告准备失败,事务管理器将向每个资源管理器发送回滚命令,并指示向应用程序提交失败。

SQL SERVER 在什么时候会选择什么样的锁的因素可能有哪些 

下面是部分SQL SERVER 中可以进行锁的资源样本

RID没有建立聚簇索引(HEAP TABLE)中标识行 ROW ID
KEY索引中的行锁,用于在可序列化事务中保护键范围
PAGE8KB的page 页,作为一个锁定的单位
EXTENT连续的8 个页面,作为锁定的单位
HoBT堆或b树。保护表中没有聚集索引的b树(索引)或堆数据页的锁。
TABLE

问题:为什么要这么多锁的类型,MYSQL innodb 不仅仅有 row lock吗?

作为一个商业数据库,在设计之初SQL SERVER 考虑了下面一个图(假设),使用低级锁(如行锁)可以降低两个事务同时请求同一数据块上的锁的概率,从而提高并发性,但使用低级锁还会增加锁的数量和管理锁所需的资源反之使用表或页锁可以降低开销,但代价是降低并发性。

SQL Server数据库引擎使用动态锁定策略来确定最经济有效的锁。数据库引擎会根据模式和查询的特性自动确定在执行查询时哪些锁是最合适的。例如,为了减少锁定的开销,优化器可以在执行索引扫描时选择索引中的页级锁定。这样做的好处也是显而易见,如果我有多行在一个PAGE中,并且都需要更改,系统会根据需要索引的资源来锁定这个PAGE,而不是一个页面里面的每个行,因为要考虑每个锁的管理,申请,释放,都是需要相关CPU 资源,内存资源的,如果能在不影响并发度的情况下,锁的粒度有效控制是有助于系统的信息的访问和修改的。

并且SQL SERVER 也是可以在表的创建,或使用中进行锁释放可以自动进行升级的设置的,你可以打开表的锁升级,或禁止掉他。

说到这里不得不说说SQL SERVER 锁的历史 SQL SERVER 7.0 之前的时候,(应该不是我出生的时候,在很久很久很久久以前久以前),SQL SERVER 是不支持 ROW 锁的,而仅仅支持 PAGE LOCK,并且一个页面是 2KB ,在 SQL SERVER 7.0,他们将SQL SERVER 变为了 8KB 的PAGE ,并且开始支持了 ROW LOCK。那到底为什么 SQL SERVER 不能做成和MYSQL 一样,仅仅支持行锁就好的数据库,为什么单库的商业数据库还是有优势的(注意这是问句)

下面是一个行锁的结构

     锁是一个64或128字节的内存结构(分别用于32位或64位机器),每个持有或请求锁的进程都有另外32或64字节。如果您需要对每一行都使用锁,并且扫描一百万行,那么您需要超过64MB的RAM来保存该进程的锁。

一个语句在一个对象上持有的锁的数量超过了一个阈值。举例目前这个阈值是5000个锁,超过就会触发esclation,如果锁分布在同一语句中的多个对象上,则不会发生锁升级——例如,一个索引中的3000个锁和另一个索引中的3000个锁,另一方面锁资源占用的内存超过启用内存的40%,那么锁会将发生升级。

那锁升级到底是好不好,回答是  呵呵, 我想你明白我的意思。为什么

当触发锁升级时,如果存在冲突锁,则会先增加更多的X锁(我想你应该是懂这个过程的),并且不同进程持有的同一表或分区上有并发的X锁,则锁升级尝试将失败。每次事务在同一对象上获得另外更多个锁时,SQL Server都会继续尝试升级锁,成功后会将SQL Server索引或堆表上的所有行锁进行释放。

可以想想这个锁升级从上到下的描述中,触发他的伴随的是大事务,占用更多的内存,系统陷入了可能繁忙的状态,那此时进行锁升级,可能会成功,可能会失败,同时成功后,锁的级别将从ROW 变成 PAGE 或者更宽泛的锁,系统将由 SHARE 变为  ONLY You  use it . 其他的事务如果访问你的资源,你是不是就不在管他们死活。

这还不是最糟糕的,锁升级会导致更多的死锁的出现,并且是莫名其妙的,看似两个根本就无关的事务,也会锁在一起,我想这时候如果你的领导来询问你的时候为什么这么多莫名其妙的死锁时,你是不是已经可以,有理有据来和他 make  clear 一下了。

这篇关于SQL SERVER 锁升级的 investigation 与 别吃黄连的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/778143

相关文章

慢sql提前分析预警和动态sql替换-Mybatis-SQL

《慢sql提前分析预警和动态sql替换-Mybatis-SQL》为防止慢SQL问题而开发的MyBatis组件,该组件能够在开发、测试阶段自动分析SQL语句,并在出现慢SQL问题时通过Ducc配置实现动... 目录背景解决思路开源方案调研设计方案详细设计使用方法1、引入依赖jar包2、配置组件XML3、核心配

MySQL数据库约束深入详解

《MySQL数据库约束深入详解》:本文主要介绍MySQL数据库约束,在MySQL数据库中,约束是用来限制进入表中的数据类型的一种技术,通过使用约束,可以确保数据的准确性、完整性和可靠性,需要的朋友... 目录一、数据库约束的概念二、约束类型三、NOT NULL 非空约束四、DEFAULT 默认值约束五、UN

MySQL 多表连接操作方法(INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL OUTER JOIN)

《MySQL多表连接操作方法(INNERJOIN、LEFTJOIN、RIGHTJOIN、FULLOUTERJOIN)》多表连接是一种将两个或多个表中的数据组合在一起的SQL操作,通过连接,... 目录一、 什么是多表连接?二、 mysql 支持的连接类型三、 多表连接的语法四、实战示例 数据准备五、连接的性

MySQL中的分组和多表连接详解

《MySQL中的分组和多表连接详解》:本文主要介绍MySQL中的分组和多表连接的相关操作,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录mysql中的分组和多表连接一、MySQL的分组(group javascriptby )二、多表连接(表连接会产生大量的数据垃圾)MySQL中的

MySQL 中的 JSON 查询案例详解

《MySQL中的JSON查询案例详解》:本文主要介绍MySQL的JSON查询的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 的 jsON 路径格式基本结构路径组件详解特殊语法元素实际示例简单路径复杂路径简写操作符注意MySQL 的 J

Spring Boot 整合 SSE的高级实践(Server-Sent Events)

《SpringBoot整合SSE的高级实践(Server-SentEvents)》SSE(Server-SentEvents)是一种基于HTTP协议的单向通信机制,允许服务器向浏览器持续发送实... 目录1、简述2、Spring Boot 中的SSE实现2.1 添加依赖2.2 实现后端接口2.3 配置超时时

Windows 上如果忘记了 MySQL 密码 重置密码的两种方法

《Windows上如果忘记了MySQL密码重置密码的两种方法》:本文主要介绍Windows上如果忘记了MySQL密码重置密码的两种方法,本文通过两种方法结合实例代码给大家介绍的非常详细,感... 目录方法 1:以跳过权限验证模式启动 mysql 并重置密码方法 2:使用 my.ini 文件的临时配置在 Wi

MySQL重复数据处理的七种高效方法

《MySQL重复数据处理的七种高效方法》你是不是也曾遇到过这样的烦恼:明明系统测试时一切正常,上线后却频频出现重复数据,大批量导数据时,总有那么几条不听话的记录导致整个事务莫名回滚,今天,我就跟大家分... 目录1. 重复数据插入问题分析1.1 问题本质1.2 常见场景图2. 基础解决方案:使用异常捕获3.

SQL中redo log 刷⼊磁盘的常见方法

《SQL中redolog刷⼊磁盘的常见方法》本文主要介绍了SQL中redolog刷⼊磁盘的常见方法,将redolog刷入磁盘的方法确保了数据的持久性和一致性,下面就来具体介绍一下,感兴趣的可以了解... 目录Redo Log 刷入磁盘的方法Redo Log 刷入磁盘的过程代码示例(伪代码)在数据库系统中,r

mysql中的group by高级用法

《mysql中的groupby高级用法》MySQL中的GROUPBY是数据聚合分析的核心功能,主要用于将结果集按指定列分组,并结合聚合函数进行统计计算,下面给大家介绍mysql中的groupby用法... 目录一、基本语法与核心功能二、基础用法示例1. 单列分组统计2. 多列组合分组3. 与WHERE结合使