Oracle 并发 锁

问题:为什么我们会收到错误"TT6002:锁请求被拒绝,因死锁而被拒绝",而我们的SQL主要是对唯一行进行插入和更新?

解决方案

回答:

  1. 当未提交的值被更新时,优化器会在执行阶段移除该旧值,而不是等待提交时。在这种情况下,用户先插入后再更新。如果更新涉及在并发事务中将索引列更新为相同值,情况也会更糟。

  2. 执行时从索引中删除需要某种下一锁来保护已删除位置,以备需要回滚时使用。

  3. 我们有两种机制可以防止随机行的下一轮锁定,具体如下:

a.对于非唯一索引,我们使用通用锁来阻止所有可序列化的扫描

b。对于唯一索引,我们插入一个带有锁

4的假行。在这种情况下,它是非唯一索引,但由于某些并发FK验证需要可序列化扫描,无法放置3.a。结果,我们被迫用下一个锁配合X锁,而X锁是死锁的根源。

以下应用端的变更有助于处理或避免死锁:

  1. 应用程序在发现死锁时应重试事务,这实际上是 Timesten 的标准和推荐做法。

  2. 通过将插入+更新合并为一个事务,避免更新未提交的值。

  3. 避免同时更新索引列到与其已存在的值相同。

这在11.2.2中被注意到,但在11.2.1和7.0中也可能出现。这种死锁在某些情况下是由于上述3级内部导致的死锁情况,因此我们正在修复该漏洞,因为遇到该问题的用户不会做出应用端的修改。这个问题在11.2.2.4.6版本中修复了159413091673896016739014的漏洞。在11.2.2.6.0版本中修复了,作为16805039的bug。这个问题在11.2.2.5.x版本中没有修复,因为11.2.2.6.0会和11.2.2.5.1差不多同时发布。

Summary

Question: Why do we get error "TT6002: Lock Request Denied Because Of Deadlock" when our SQL is mostly inserts and updates on unique rows?

Solution

Answer:

  1. When an uncommitted value is updated, optimizer will remove the old value from the index during execution phase instead of waiting until commit time. In this case, user did an insert followed by an update. Situation can also be worse if the update involves updating the indexed column to the same value in concurrent transactions.

  2. Delete from the index at execution will require some kind of next lock to protect the deleted position in case a rollback is needed.

  3. We have two mechanisms to prevent next locks on random row as follows:

a. For non-unique index, we use a general lock to prevent all serializable scan

b. For unique index, we insert a dummy row with a lock

  1. In this case, it was a non-unique index but 3.a could not be placed because some concurrent FK verification need serializable scan. As a result, we were forced to use next lock with X lock and X lock is a source of deadlock.

The following application side changes could help handle or avoid the deadlock:

  1. Application should retry the transaction when it catches that a deadlock occurs, this is actually standard and recommended practice for Timesten.

  2. Avoid updating of uncommitted value by combining insert+update into one transaction.

  3. Avoid doing concurrent updates of indexed column to the same value as it already is.

This was noticed in 11.2.2, but can occur in 11.2.1 and 7.0. This deadlock in some cases is the result of internally caused deadlock situation described above as 3, so for that reason we are fixing the bug since the particular users that encountered it would not make application side changes. This was fixed in 11.2.2.4.6 in bugs 15941309, 16738960, and 16739014. It is fixed in 11.2.2.6.0 as bug 16805039. This was not fixed in the 11.2.2.5.x release since 11.2.2.6.0 will come out around the same time as 11.2.2.5.1.

相关推荐
三十..1 小时前
Redis 核心原理与高可用架构实践
运维·数据库·redis
这个DBA有点耶1 小时前
索引优化深潜(下):索引合并、ICP 与索引设计的实战法则
数据库·mysql·架构
努力努力再努力wz2 小时前
【内存管理与高并发内存池系列】从 mmap 到 malloc:文件映射、匿名映射与 glibc 内存分配机制详解
linux·c语言·数据结构·数据库·c++·qt·链表
JdSnE27zv2 小时前
Qt 操作SQLite数据库
数据库·qt·sqlite
tedcloud1232 小时前
HyperFrames部署教程:用HTML生成MP4视频
前端·数据库·人工智能·html·音视频
布朗克1682 小时前
25 IO流高级操作——序列化、NIO与Files工具类
java·数据库·io·nio
阿演2 小时前
DataDjinn 新版本更新:新增 Oracle 支持,查询窗口、表预览和连接树继续打磨
数据库·oracle·ai编程·数据库连接工具
lixora3 小时前
Oracle 11g Active Data Guard Go 自动化部署工具 v1.0
数据库·oracle
Nturmoils3 小时前
自增主键别只会 auto_increment,先把值从哪来讲清楚
数据库·后端
叶小鸡3 小时前
Java 篇-项目实战-AI 天机学堂(从 0 到 1)-day5
数据库·redis·缓存