Oracle lock 锁机制 总结

对于lock这个词,要从两方面理解,首先,它代表的是一种控制机制;其次,在这个机制中有个一成员也叫lock。

从机制的角度看,lock框架包括3个组件:resource structure(资源);lock structure(锁);enqueue(排队机制)

resource和lock是数据结构,enqueue是使用的算法。

下面看一下lock的组成结构图:

resource structure:oracle对于每个需要进行“并发控制”的资源,都在SGA中用一个数据结构来描述它,这个数据结构就叫做resource structure。这个数据结构中有3个与“并发控制” 有关的成员:Owner ,Waiter , Converter这是三个指针,分别指向3个由lock structrue组成的链表。

lock structure:每当进程要访问共享资源时,必须先锁定该资源,这个动作实际就是从内存中申请一个lock sturture 在其中记录“锁模式,进程ID”等重要信息,然后看是否能够立即获得资源的访问权,如果不能,则把这个lock structure 挂到resource structure 的waiter 链表中;

如果能够获得,则把lock structure 挂到resource structure的owene链表中。上面说的resource structure 中的owner ,waiter converter 三个成员就是指向由lock struncture组成的链表指针。

enqueue算法:lock使用的是enqueue算法,可以理解为“先入先出队列”。如果进程的锁定请求不能满足,该进程的lock structure 就被加到waiter 链表的末端。当占用进程释放锁时,会检查waiter 和 converter 队列,把锁分配给最先进入队列的请求者。

converter 和 waiter 两个都是等待队列,二者的用法有细微区别:如果两个操作先后需要两种不同模式的锁,比如先是shared mode然后是 exclusive mode,则进程会先请求shared mode ,获得后lock structure 会挂载owner队列上,当需要exclusive mode 锁时,请求进程必须先释放shared mode的锁,然后再次申请exclusive mode 的锁,但是这可能无法立即获得。这时,这个请求就会被挂在converter队列下,converter队列会优先于waiter队列被处理。

可以从v$lock视图中看到这些lock信息,并且还可以根据v$lock视图的lmode 和request mode 判断出谁是owner,waiter 和 converter:

  lmode>0  ,  request=0   ==> owner

  lmode=0  ,  request>0   ==> waiter(acquirer)

  lmode>0  ,  request >0  ==> converter

数据记录的行级锁

在讲到lock的上文中已经提到,这种控制机制需要resource,lock联众数据结构,但是需要内存分配lock数据结构,对于粗粒度(可以理解为文件很大)或者数量有限的资源,使用这种机制还可以接受,因为分配的内存并不多。但是对于表的数据记录,动辄几百G 的表,每个记录如果都分配一个resource 和lock 数据结构对,无论从内存需求还是维护开销上都是一个噩梦。所以,对于数据记录这种细粒度的资源,oracle使用的是行级锁(row level  lock)。行级锁的实现原理如下图,左边代表数据块,右边代表回滚段(undo
segment):

具体数据块的结构详细介绍参考我之前的博客:http://blog.csdn.net/changyanmanman/article/details/7076935

我们现在只需要知道,数据块内存储的是一条条的用户记录,用户记录也是按照一定的格式保存的,每条记录可以分成 记录头 和 记录体 两部分。记录头中是描述信息,比如列宽度,和事务有关的是ITL Entry pointer字段。整个行级锁供涉及以下4中数据结构,这是我们要研究的重点:

1)ITL:这个比较熟悉,用于记录哪些事务修改了这个数据块的内容,可以把他想象成一个表格,每个表格对应一个事务,包括事务号,事务是否提交等重要信息。

2)记录头ITL索引:每条记录的记录头部有一个字段,用于记录ITL表项号,可以看做是指向ITL表的指针,如果一个进程来访问记录,那就先得根据这个指针去ITL中看看事务已经提交了没啊。。前一个事务是用的什么锁啊。。等待

3)TX锁:这个锁代表一个事务,这个锁属于上文讲到的lock机制,有resource structure , lock structure , enqueue 算法。

4)TM锁:这个锁也属于前面讲到的lock机制,用于保护对象(表,视图等)的定义不被修改。

下面我们详细说一下运行机制:

 当一个事务开始时,必须先申请一个TX锁,注意 这种锁保护的资源是回滚段,回滚数据块。因此这个申请也就意味着:用户必须先申请到一个回滚段资源后才能开始一个事务,才能执行语句修改数据。

申请到回滚段资源后,用户事务就可以修改数据了。在修改数据表的记录时,需要遵循下面的操作顺序:

1、首先获得这个表的TM锁,这个锁用于保护事务执行过程中其他用户不能修改表结构(但是可以修改表内的数据)。

2、事务修改某个数据块中的记录时,首先要在数据块块头的ITL表中申请一个空闲表项,并且在其中记录事务号,实际就是记录这个事务要使用的回滚段地址。

3、事务修改该数块中的某条记录时,会设置该记录头部的ITL索引指向上一步申请到的表项。然后再修改记录的内容,修改前现在回滚段对记录修改该钱的状态做一个拷贝,然后才能修改该数据记录,这个拷贝用于以后的回滚,恢复,或者一致性读。

4、当其他用户并发修改该这条记录时,会根据记录头的ITL索引读取ITL表项内容。查看这个事务石头已经提交。

5、如果没有提交,则这个用户的TX锁会等待前一个用户的TX锁的释放。

从上面的工作机制可以看出,无论一个事务修改多少个表的多少条记录,盖世五真正需要的只是一个TX锁,每个表一个的TM锁,内存开销非常小。而所谓的行级锁,其实只是数据块头,数据记录的一些字段,不会消耗额外的内存资源。

锁模式:

在介绍锁模式之前,我们先要了解一个概念,oracle中的对象分类:oracle中的对象分为两种:简单对象(simple object) 和 复合对象(compound object)  数据表是典型的复合对象,表中的每条记录都是典型的简单对象。

对于简单对象,只有三种锁模式:null ,share , exclusive

对于符合对象,除了这种外,还有三种 :sub-shared,sub-exclusive,share-sub-exclusive

这6种模式的保护能力从高到低的顺序如图:

之所以吧对象划分成联众类型,也是从性能角度考虑。比如oracle提供lock table语句允许对表加锁,在锁表之前,oracle必须确保没有其他用户操作表中的数据 。加入之后行级锁,就会有这样一个问题,oracle必须对表中的所有记录做一个遍历,才能确定是否有不兼容的锁模式存在,而且还得对扫描过的记录加锁,以阻止其他用户操作,对系统并发能力影响很大。通过吧对象按简单对象,复合对象分类,并使用不同锁,对想能有很大提升。

下图列举了常见sql语句需要的锁模式已经模式键的兼容关系,其中第一列:sql操作,第二列:(mode of tbale lock)指的是左边也就是第一列语句加在表上的锁模式。 第三列:(lock modes permitted)以及下面的5个小列,分别说明不同模式间的兼容关系。

下面我们通过几个语句来学习这个表:

select * from a;

对照表的第一行,这个语句不会在表a上添加任何模式的锁,兼容其他所有模式的锁。也就是说用户a在执行select操作时,其他用户可以执行update,insert,delete,lock table等所有操作。

insert a values (...) / update/delete

看表的第2到4行,由第二列可见这3个语句都会在表a上添加RX模式的锁(sub-exclusive)这个模式的锁能兼容RS,还有自己同样的RX锁,从该表的第二列还可以看到insert、update、delete 这3个语句是RX模式的锁,而select ....for update . lock table ...in row share mode 是RS模式的锁,也就是说当执行 insert 、update 、delete 语句时,其他用户只能执行 上述5个操作,而不能执行其他操作。

但是这里所说的允许操作,只是只在表级别(或者说复合对象级别)允许操作,单最终操作是否能成功还要根据行级锁。这也是update、delete、select ...for update 语句后面 Y*  加星号的原因。

下面我们以library cache 为例,看一下null 锁模式,我们知道这个library cache 中存放的是 所有的sql语句,执行计划,包 等对象,以及这些对象所引用的对象(比如执行计划引用的对象表) 当一个语句或包编译时,这个语句引用的对象都会加上一个 library cache lock ,而执行时,所有这些引用对象都要背加一个library
cache pin。 以保证在语句执行过程中,应用对象的结构不会被修改。

      编译完成后,加在引用对象上的library cache lock 会由原来的share 或者 exclusive模式 变成 null 模式,null模式的library cache lock 就相当于一个触发器,当这些对象的定义被修改时,引用他的对象将被置为无效,必须重新编译。

     我们举个例子:比如:“select * from a ”编译之后,这个语句的执行计划对象会在a上加一个null模式的library cache lock,这个null模式的lock相当于一个触发器,以后如果对表a的结构做了修改,比如增加一个字段,这个触发器就会导致“select * from a ”这个语句的执行计划失效。如果再次执行这个语句,旧的执行计划就不能再被使用,必须重新编译产生新的执行计划。

       在RAC环境中,要知道,每个节点都可能有表a的引用对象,无论在任何一个节点上对a的结构进行修改,都需要把所有节点上应用a的对象(即各个节点library cache中执行计划)值为无效,因此除了传统的library cache lock之外,每个节点上的LCK0进程,会对本实例library cache中的对象加一个共享模式(shared-mode)的IV(Invalidation 无效) Instance Lock,如果某个用户想要修改对象定义,必须获得一个exclusive模式的iv锁,这会通知本地的LCK0释放shared-mode锁,本地LCK0在释放这个锁之前,会通知其他节点上的LCK0,其他节点的LCK0收到这个消息后,就会将本地library
cache中 的所有相关对象值为失效,这种机制是一种广播机制,这些通信过程是通过实例的LMD进程完成的。

对于cache fusion 资源。比如典型额buffer cache 的数据块数量更多,修改更加密集,如果每次数据块的修改都在集群内发送广播,这种方式显得太低效了。因此,对于数据块这种资源,oracle采用的是cache fusion机制,这种机制使用的是pcm lock。关于pcm lock 。。。。它有三种模式shared ,exclusive,null。。对于SGA中的数据块,oracle用另一个属于来描述加在每个数据块上的锁类型——数据块状态(buffer state)

时间: 2024-11-19 02:22:51

Oracle lock 锁机制 总结的相关文章

介绍了Oracle数据库锁的种类及研究

本文通过对Oracle数据库锁机制的研究,首先介绍了Oracle数据库锁的种类,并描述了实际应用中遇到的与锁相关的异常情况,特别对经常遇到的由于等待锁而使事务被挂起的问题进行了定位及解决,并对死锁这一比较严重的现象,提出了相应的解决方法和具体的分析过程. 数据库是一个多用户使用的共享资源.当多个用户并发地存取数据时,在数据库中就会产生多个事务同时存取同一数据的情况.若对并发操作不加控制就可能会读取和存储不正确的数据,破坏数据库的一致性. 加锁是实现数据库并发控制的一个非常重要的技术.当事务在对某

利用Oracle执行计划机制提高查询性能

oracle|性能|执行 消耗在准备利用Oracle执行计划机制提高查询性能新的SQL语句的时间是Oracle SQL语句执行时间的最重要的组成部分.但是通过理解Oracle内部产生执行计划的机制,你能够控制Oracle花费在评估连接顺序的时间数量,并且能在大体上提高查询性能. 准备执行SQL语句 当SQL语句进入Oracle的库缓存后,在该语句准备执行之前,将执行下列步骤: 1) 语法检查:检查SQL语句拼写是否正确和词序. 2) 语义分析:核实所有的与数据字典不一致的表和列的名字. 3) 轮

理解oracle锁和闩(2)锁机制概述

锁(lock)是一种防止多个事务访问同一资源时产生破坏性的相互影响的机制.通常,高并发数据库需要利用锁机制解决数据并发访问.一致性及完整性问题. 前面提到的资源(resource)大致可以分为两类: ● 用户对象:例如表及数据行 ● 对用户透明的系统对象:例如内存中的共享数据结构.数据字典中的信息 任何 SQL 语句执行时 Oracle 都隐式地对 SQL 所需的锁进行管理,因此用户无需显式地对资源加锁.Oracle 默认采用的锁机制能尽可能地减小对数据访问的限制,在保证数据一致性的同时实现高度

Oracle数据完整性和锁机制简析_oracle

本课内容属于Oracle高级课程范畴,内容略微偏向理论性,但是与数据库程序开发和管理.优化密切相关:另外本课的部分内容在前面章节已经涉及,请注意理论联系实际. 事务 事务(Transaction)从 通讯的角度看:是用户定义的数据库操作序列,这些操作要么全做.要么全不做,是不可分割的一个工作单元.事务控制语句称为TCL,一般包括Commit和Rollback. 事务不是程序,事务和程序分属两个概念.在RDBMS中,一个事务可以有一条SQL语句.一组SQL语句或者整个程序:一个应用程序又通常包含多

ORACLE基础之oracle锁(oracle lock mode)详解

ORACLE里锁有以下几种模式: 0:none 1:null 空 2:Row-S 行共享(RS):共享表锁,sub share  3:Row-X 行独占(RX):用于行的修改,sub exclusive  4:Share 共享锁(S):阻止其他DML操作,share 5:S/Row-X 共享行独占(SRX):阻止其他事务操作,share/sub exclusive  6:exclusive 独占(X):独立访问使用,exclusive 1.oracle提供的所类型可以根据v$lock_type

ORACLE锁机制深入理解_oracle

数据库是一个多用户使用的共享资源.当多个用户并发地存取数据时,在数据库中就会产生多个事务同时存取同一数据的情况.若对并发操作不加控制就可能会读取和存储不正确的数据,破坏数据库的一致性. 加锁是实现数据库并发控制的一个非常重要的技术.当事务在对某个数据对象进行操作前,先向系统发出请求,对其加锁.加锁后事务就对该数据对象有了一定的控制,在该事务释放锁之前,其他的事务不能对此数据对象进行更新操作. 在数据库中有两种基本的锁类型:排它锁(Exclusive Locks,即X锁)和共享锁(Share Lo

MSSQL与Oracle数据库事务隔离级别与锁机制对比_oracle

一,事务的4个基本特征 Atomic(原子性): 事务中包含的操作被看做一个逻辑单元,这个逻辑单元中的操作要 么全部成功,要么全部失败. Consistency(一致性): 只有合法的数据可以被写入数据库,否则事务应该将其回滚到最初 状态. Isolation(隔离性): 事务允许多个用户对同一个数据进行并发访问,而不破坏数据的正 确性和完整性.同时,并行事务的修改必须与其他并行事务的修改 相互独立. Durability(持久性): 事务结束后,事务处理的结果必须能够得到固化. 以上属于废话

深入JVM锁机制1-synchronized

[本文转载于深入JVM锁机制1-synchronized] 目前在Java中存在两种锁机制:synchronized和Lock,Lock接口及其实现类是JDK5增加的内容,其作者是大名鼎鼎的并发专家Doug Lea.本文并不比较synchronized与Lock孰优孰劣,只是介绍二者的实现原理.    数据同步需要依赖锁,那锁的同步又依赖谁?synchronized给出的答案是在软件层面依赖JVM,而Lock给出的方案是在硬件层面依赖特殊的CPU指令,大家可能会进一步追问:JVM底层又是如何实现

[转]深入JVM锁机制1-synchronized

转自:http://blog.csdn.net/chen77716/article/details/6618779    目前在Java中存在两种锁机制:synchronized和Lock,Lock接口及其实现类是JDK5增加的内容,其作者是大名鼎鼎的并发专家Doug Lea.本文并不比较synchronized与Lock孰优孰劣,只是介绍二者的实现原理.    数据同步需要依赖锁,那锁的同步又依赖谁?synchronized给出的答案是在软件层面依赖JVM,而Lock给出的方案是在硬件层面依赖