SQL Server死锁
SQL Server死锁
- 多个事务之间互相等待对方的资源,导致这些事务永久等待
- 注意是永久等待,而非长事务
死锁的4个条件
- 互斥条件(Mutual exclusion):资源不能被共享,只能由一个进程使用。
- 请求与保持条件(Hold and wait):已经得到资源的进程可以再次申请新的资源。
- 非剥夺条件(No pre-emption):已经分配的资源不能从相应的进程中被强制地剥夺。
- 循环等待条件(Circular wait):系统中若干进程组成环路,该环路中每个进程都在等待相邻进程正占用的资源。
事务隔离级别
- READ UNCOMMITTED(SET TRANSACTION ISOLATION LEVEL READ UNCOMMITTED)
- READ COMMITTED(SET TRANSACTION ISOLATION LEVEL READ COMMITTED)
- REPEATABLE READ(SET TRANSACTION ISOLATION LEVEL REPEATABLE READ)
- SERIALIZABLE(SET TRANSACTION ISOLATION LEVEL SERIALIZABLE)
- SNAPSHOT(SET TRANSACTION ISOLATION LEVEL SNAPSHOT)
验证
- 初始数据
- 默认隔离等级下,更新事务对事务外SELECT语句的影响
- 在sql server ssms中分别打开两个连接,分别模拟事务以及普通SELECT
- 下面这个事务只有begin,没有COMMIT或者ROLLBACK,目的是模拟并发时多个session同时运行效果

- 执行上面这个事务
- 接着在另外一个session中分别查询下面这些select
/*example 1*/
SELECT * FROM [dbo].[customer]
SELECT * FROM [dbo].[customer] WHERE id=1
SELECT * FROM [dbo].[customer] WHERE id=2
SELECT * FROM [dbo].[customer](NOLOCK) WHERE id=2分别执行后的结果(结果里空白的为阻塞状态)

- 然后在事务session中执行rollback命令
- 在普通SELECT session中运行下面语句
SELECT * FROM [dbo].[customer]

- 可以看到,这里又变成了aaa(在NOLOCK查询时这个值为Changed)
- 结论
- 在默认隔离级别下(READ COMMITTED),对于UPDATE过的记录,在事务范围外默认会被阻塞住
- 同样适用于REPEATABLE READ以及SERIALIZABLE,但不适用于READ UNCOMMITTED
- 同样适用于DELETE/INSERT记录
- 如果在事务外部进行查询的查询条件不牵涉到事务内的更新记录,则外部查询不会被阻塞
- 比如WHERE id=2会被阻塞
- 比如WHERE id=3不会被阻塞
- 比如WHERE id between 1 AND 3会被阻塞
- 其他线下证明
- 例子中是用了pkid来做where条件,但是用其他条件也同样适用
- 例子中的pkid有index,但是没有index的字段同样也适用
- 在默认隔离级别下(READ COMMITTED),对于UPDATE过的记录,在事务范围外默认会被阻塞住
- SERIALIZABLE隔离等级下,对事务外的SELECT/UPDATE/DELETE影响
- 初始数据为


- 执行上面这个更改为serializable的事务
- 然后在这个事务外部执行下面测试



- 结论
- 如果某事务是用了serializable隔离级别,则外部就无法insert任何记录到相应的表中
- 无论insert的字段值是否落在where条件内或者外
- 外部的UPDATE行为:同普通等级
- 实际上DELETE行为也是
- 如果某事务是用了serializable隔离级别,则外部就无法insert任何记录到相应的表中
- 默认隔离级别中插入的记录,对于外部查询的影响

- 执行下面这些测试


- 结论
- 如果某事务是用了read committed隔离级别,则外部默认无法看到这些新增的记录
- 同样适用于repeatable read以及serilizable隔离级别
- 使用了NOLOCK关键字的查询语句能看到脏数据
- 如果某事务是用了read committed隔离级别,则外部默认无法看到这些新增的记录
- 默认隔离级别下事务死锁例子
- 初始数据

- 我们在2个session中同时执行2个事务,如下

- 结论
- 互相等待资源的释放、同时又持有现有资源,就会造成死锁
- 上面例子中被kill掉的事务,选择牺牲事务的算法,ms称:正常情况下,SQL Server会把它认为取消或回滚代价最小的连接作为默认的死锁牺牲品
- 算了,不深入了
- 知道还有个非正常情况 - 死锁优先级,后续
- 事务优先级 - SET DEADLOCK_PRIORITY(http://msdn.microsoft.com/en-us/library/ms186736.aspx)

- 也可以如下

- 结论
- SET DEADLOCK_PRIORITY可以放在SESSION的任何地方,无论是事务外还是内、前还是后
- sqlserver探测到死锁后,会根据session的deadlock优先级来kill
- LOW/NORMAL/HIGH
- 分别代表
- -5/0/5
- 分别代表
- 也可以使用数字
- -10直到10
- SET DEADLOCK_PRIORITY可以放在SESSION的任何地方,无论是事务外还是内、前还是后
避免死锁的建议
- 把SELECT放在事务范围外
- 将多个事务合并为一个事务
- SELECT加With(NOLOCK)提示
- 降低事务隔离级别, 比如READ UNCOMMITTED
- 使用基于行版本控制的隔离级别
SQL Server死锁的更多相关文章
- SQL Server死锁的解除方法
如果想要查出SQL Server死锁的原因,下面就教您SQL Server死锁监控的语句写法,如果您对此方面感兴趣的话,不妨一看. 下面的SQL语句运行之后,便可以查找出SQLServer死锁和阻塞的 ...
- Update导致SQL Server死锁的典型方法(转载)
此文为转载文章,描述的很好,没有验证过. 最近遇到了一个看上去很奇怪,分析起来很有意思的死锁问题.这个死锁看上去难以理解.而分析过程中,又使用了很多分析SQL Server死锁的典型方法.记录下来整个 ...
- SQL Server死锁的分析、处理与预防
1.基本原理 所谓“死锁”,在操作系统的定义是:在一组进程中的各个进程均占有不会释放的资源,但因互相申请被其他进程所站用不会释放的资源而处于的一种永久等待状态. 定义比较抽象,下图可以帮助你比较直观的 ...
- SQL Server死锁日志各字段含义
使用跟踪标记 1204 --打开跟踪标记 DBCC TRACEON (1204,-1) --关闭跟踪标记 DBCC TRACEOFF (1204,-1) 处于死锁状态时,跟踪标记 1204 在等待的线 ...
- SQL Server 死锁概念和分析
锁的概念 锁是什么 锁是数据库中在并发操作情形下保护资源的机制.通常(具体要看锁兼容性)只有锁的拥有者才能对被锁的资源进行操作,从而保证数据一致性. 锁的概念可分为几部分 锁资源(锁住什么) 锁模式( ...
- SQL Server 死锁的告警监控
今天这篇文章总结一下如何监控SQL Server的死锁,其实以前写过MS SQL 监控错误日志的告警信息,这篇文章着重介绍如何监控数据库的死锁,当然这篇文章不分析死锁产生的原因.以及如何解决死锁.死锁 ...
- sql server 死锁排查
记得以前客户在使用软件时,有偶发出现死锁问题,因为发生的时间不确定,不好做问题的重现,当时解决问题有点棘手了. 现总结下查看死锁的常用二种方式: 第一种是图形化监听: sqlserver --> ...
- SQL Server死锁中的会话隔离级别为序列化(Serializable)实验测试
最近在分析SQL Server的死锁时,发现一个比较有意思的现象,发现死锁当中一个会话的隔离级别为序列化(Serializable),这个是让人比较奇怪的地方,我们知道SQL Server数据库的默认 ...
- SQL Server死锁诊断--同一行数据在不同索引操作下引起的死锁
死锁概述 对于数据库中出现的死锁,通俗地解释就是:不同Session(会话)持有一部分资源,并且同时相互排他性地申请对方持有的资源,然后双方都得不到自己想要的资源,从而造成的一种僵持的现象.当然,在任 ...
随机推荐
- android 区分wifi是5G还是2.4G
http://bbs.csdn.net/topics/391033966?page=1 我一开始看这帖子,找不到答案,为了后来的人,我来回复吧.WifiManager wifiManager = (W ...
- opencv学习_15 (利用cmake查看opencv的源码)
当我们有时想查看opencv自带的函数的源代码,比如函数cvCreateImage, 此时我们选中cvCreateImage, 点击鼠标右键->转到定义,我们会很惊讶的发现为什么只看到了cvCr ...
- 利用ManualResetEvent来来控制异步调用的打印的线程的暂停和恢复(转)
利用ManualResetEvent来来控制异步调用的打印的线程的暂停和恢复 打印过程可能很长,这时候有可能需要暂停下来做一些事情,然后回来继续接着打印 打印过程中有2个线程:一个是程序运行的主线程, ...
- DBAPI部署
1.添加源 sudo rpm -Uvh http://download.fedoraproject.org/pub/epel/6/x86_64/epel-release-6-8.noarch.rpm ...
- 盘点国内网站常用的一些 CDN 公共库加速服务
CDN公共库是指将常用的JS库存放在CDN节点,以方便广大开发者直接调用.与将JS库存放在服务器单机上相比,CDN公共库更加稳定.高速.一 般的CDN公共库都会包含全球所有最流行的开源JavaScri ...
- [Xamarin] 關於Internal Storage ,存取App內部使用資料 (转帖)
最近在開發App,會使用到必須要處理一些App所使用的資料,上網路查一下Android 得作法,包含我自己也實作了一下,可能是因為對Java || Android 不是很孰悉,常常錯在 java.la ...
- Asp.Net Web API 2第十四课——Content Negotiation(内容协商)
前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok/p/3446289.html 本文描述ASP.NET W ...
- 基于OWIN WebAPI 使用OAuth授权服务【客户端模式(Client Credentials Grant)】
适应范围 采用Client Credentials方式,即应用公钥.密钥方式获取Access Token,适用于任何类型应用,但通过它所获取的Access Token只能用于访问与用户无关的Open ...
- Arcgis for Javascript 出现“init.js->TypeError: f is not a function”
环境 采用离线JS包,版本为v3.8 问题描述 在为map添加了 app.map.on("pan-start", this.showHandBeignPan()); 在拖动地图的时 ...
- Kali Linux Web 渗透测试视频教程— 第四课 google hack 实战
Kali Linux Web 渗透测试— 第四课 google hack 实战 文/玄魂 目录 shellKali Linux Web 渗透测试— 第四课 google hack 实战 课程目录 Go ...
