大橙子网站建设,新征程启航
为企业提供网站建设、域名注册、服务器等服务
如果当前有用户在对某行数据进行修改登操作,oracle会在这行数据上添加行级锁,期间,所有用户对该行数据只能查询,不可修改,如果比如说执行update操作,需等待该修改操作事务提交或者回滚之后,才行。
创新互联建站专注为客户提供全方位的互联网综合服务,包含不限于做网站、成都网站建设、掇刀网络推广、成都微信小程序、掇刀网络营销、掇刀企业策划、掇刀品牌公关、搜索引擎seo、人物专访、企业宣传片、企业代运营等,从售前售中售后,我们都将竭诚为您服务,您的肯定,是我们最大的嘉奖;创新互联建站为所有大学生创业者提供掇刀建站搭建服务,24小时服务热线:18980820575,官方网址:www.cdcxhl.com
此文章主要是对Oracle数据库锁机制的详细研究 首先我们要介绍的是Oracle数据库锁的类型 同时也阐述 在实际应用中我们经常会遇到的与锁相关的异常情况 特别对经常遇到的由于等待锁而使事务被挂起的问题进行了定位及解决 并对死锁这一比较严重的现象 提出了相应的解决方法和具体的分析过程
数据库是一个多用户使用的共享资源 当多个用户并发地存取数据时 在数据库中就会产生多个事务同时存取同一数据的情况 若对并发操作不加控制就可能会读取和存储不正确的数据 破坏数据库的一致性
加锁是实现数据库并发控制的一个非常重要的技术 当事务在对某个数据对象进行操作前 先向系统发出请求 对其加锁 加锁后事务就对该数据对象有了一定的控制 在该事务释放锁之前 其他的事务不能对此数据对象进行更新操作
在数据库中有两种基本的锁类型 排它锁(Exclusive Locks 即X锁)和共享锁(Share Locks 即S锁) 当数据对象被加上排它锁时 其他的事务不能对它读取和修改 加了共享锁的数据对象可以被其他事务读取 但不能修改 数据库利用这两种基本的锁类型来对Oracle数据库的事务进行并发控制
在实际应用中经常会遇到的与锁相关的异常情况 如由于等待锁事务被挂起 死锁等现象 如果不能及时地解决 将严重影响应用的正常执行 而目前对于该类问题的解决缺乏系统化研究和指导 本文在总结实际经验的基础上 提出了相应的解决方法和具体的分析过程
Oracle数据库的锁类型
根据保护的对象不同 Oracle数据库锁可以分为以下几大类 DML锁(data locks 数据锁) 用于保护数据的完整性 DDL锁(dictionary locks 字典锁) 用于保护数据库对象的结构 如表 索引等的结构定义 内部锁和闩(internal locks and latches) 保护数据库的内部结构
DML锁的目的在于保证并 *** 况下的数据完整性 本文主要讨论DML锁 在Oracle数据库中 DML锁主要包括TM锁和TX锁 其中TM锁称为表级锁 TX锁称为事务锁或行级锁
当Oracle执行DML语句时 系统自动在所要操作的表上申请TM类型的锁 当TM锁获得后 系统再自动申请TX类型的锁 并将实际锁定的数据行的锁标志位进行置位 这样在事务加锁前检查TX锁相容性时就不用再逐行检查锁标志 而只需检查TM锁模式的相容性即可 大大提高了系统的效率
TM锁包括了SS SX S X等多种模式 在Oracle数据库中用 - 来表示 不同的SQL操作产生不同类型的TM锁 如表 所示
在数据行上只有X锁(排他锁) 在 Oracle数据库中 当一个事务首次发起一个DML语句时就获得一个TX锁 该锁保持到事务被提交或回滚 当两个或多个会话在表的同一条记录上执行DML语句时 第一个会话在该条记录上加锁 其他的会话处于等待状态 当第一个会话提交后 TX锁被释放 其他会话才可以加锁
当Oracle数据库发生TX锁等待时 如果不及时处理常常会引起Oracle数据库挂起 或导致死锁的发生 产生ORA 的错误 这些现象都会对实际应用产生极大的危害 如长时间未响应 大量事务失败等
TX锁等待的分析
在介绍了有关地Oracle数据库锁的种类后 下面讨论如何有效地监控和解决锁等待现象 及在产生死锁时如何定位死锁的原因
监控锁的相关视图 数据字典是Oracle数据库的重要组成部分 用户可以通过查询数据字典视图来获得数据库的信息 和锁相关的数据字典视图如表 所示
TX锁等待的监控和解决在日常工作中 如果发现在执行某条SQL时数据库长时间没有响应 很可能是产生了TX锁等待的现象 为解决这个问题 首先应该找出持锁的事务 然后再进行相关的处理 如提交事务或强行中断事务
死锁的监控和解决在数据库中 当两个或多个会话请求同一个资源时会产生死锁的现象 死锁的常见类型是行级锁死锁和页级锁死锁 Oracle数据库中一般使用行级锁 下面主要讨论行级锁的死锁现象
当Oracle检测到死锁产生时 中断并回滚死锁相关语句的执行 报ORA 的错误并记录在Oracle数据库的日志文件alertSID log中 同时在user_dump_dest下产生了一个跟踪文件 详细描述死锁的相关信息
在日常工作中 如果发现在日志文件中记录了ora 的错误信息 则表明产生了死锁 这时需要找到对应的跟踪文件 根据跟踪文件的信息定位产生的原因
如果查询结果表明 死锁是由于bitmap索引引起的 将IND_T_PRODUCT_HIS_STATE索引改为normal索引后 即可解决死锁的问题
表 Oracle的TM锁类型
锁模式 锁描述 解释 SQL操作
none
NULL 空 Select
SS(Row S) 行级共享锁 其他对象只能查询这些数据行 Select for update Lock for update Lock row share
SX(Row X) 行级排它锁 在提交前不允许做DML操作 Insert Update Delete Lock row share
S(Share) 共享锁 Create index Lock share
SSX(S/Row X) 共享行级排它锁 Lock share row exclusive
lishixinzhi/Article/program/Oracle/201311/18509
1\x0d\x0a如何锁一个表的某一行\x0d\x0a\x0d\x0aSETTRANSACTION\x0d\x0aISOLATIONLEVELREADUNCOMMITTED\x0d\x0a\x0d\x0aSELECT*FROMtableROWLOCKWHEREid=1\x0d\x0a\x0d\x0a2锁定数据库的一个表\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(HOLDLOCK)\x0d\x0a\x0d\x0a加锁语句:\x0d\x0asybase:\x0d\x0aupdate表setcol1=col1where1=0\x0d\x0a;\x0d\x0aMSSQL:\x0d\x0aselectcol1from表(tablockx)\x0d\x0awhere\x0d\x0a1=0\x0d\x0a;\x0d\x0aoracle:\x0d\x0aLOCKTABLE表INEXCLUSIVEMODE;\x0d\x0a\x0d\x0a加锁后其它人不可操作,直到加锁用户解锁,用commit或rollback解锁\x0d\x0a\x0d\x0a几个例子帮助大家加深印象\x0d\x0a\x0d\x0a设table1(A,B,C)\x0d\x0aABC\x0d\x0aa1b1c1\x0d\x0aa2b2c2\x0d\x0aa3b3c3\x0d\x0a\x0d\x0a1)排它锁\x0d\x0a新建两个连接\x0d\x0a在第一个连接中执行以下语句\x0d\x0abegintran\x0d\x0aupdatetable1\x0d\x0a\x0d\x0aset\x0d\x0aA='aa'\x0d\x0awhereB='b2'\x0d\x0awaitfordelay\x0d\x0a'00:00:30'--等待30秒\x0d\x0acommittran\x0d\x0a\x0d\x0a在第二个连接中执行以下语句\x0d\x0abegintran\x0d\x0aselect*fromtable1\x0d\x0a\x0d\x0awhereB='b2'\x0d\x0acommittran\x0d\x0a\x0d\x0a若同时执行上述两个语句,则select查询必须等待update执行完毕才能执行即要等待30秒\x0d\x0a\x0d\x0a2)共享锁\x0d\x0a在第一个连接中执行以下语句\x0d\x0abegintran\x0d\x0aselect*fromtable1\x0d\x0aholdlock\x0d\x0a-holdlock人为加锁\x0d\x0awhereB='b2'\x0d\x0awaitfordelay\x0d\x0a'00:00:30'--等待30秒\x0d\x0acommittran\x0d\x0a\x0d\x0a在第二个连接中执行以下语句\x0d\x0abegintran\x0d\x0aselectA,C\x0d\x0afrom\x0d\x0atable1\x0d\x0awhereB='b2'\x0d\x0aupdatetable1\x0d\x0a\x0d\x0aset\x0d\x0aA='aa'\x0d\x0awhereB='b2'\x0d\x0acommittran\x0d\x0a\x0d\x0a若同时执行上述两个语句,则第二个连接中的select查询可以执行\x0d\x0a而update必须等待第一个事务释放共享锁转为排它锁后才能执行\x0d\x0a即要等待30秒\x0d\x0a\x0d\x0a3)死锁\x0d\x0a增设table2(D,E)\x0d\x0aDE\x0d\x0ad1e1\x0d\x0ad2e2\x0d\x0a\x0d\x0a在第一个连接中执行以下语句\x0d\x0abegintran\x0d\x0aupdatetable1\x0d\x0a\x0d\x0aset\x0d\x0aA='aa'\x0d\x0awhereB='b2'\x0d\x0awaitfordelay\x0d\x0a'00:00:30'\x0d\x0aupdatetable2\x0d\x0a\x0d\x0aset\x0d\x0aD='d5'\x0d\x0awhereE='e1'\x0d\x0acommittran\x0d\x0a\x0d\x0a在第二个连接中执行以下语句\x0d\x0abegintran\x0d\x0aupdatetable2\x0d\x0a\x0d\x0aset\x0d\x0aD='d5'\x0d\x0awhereE='e1'\x0d\x0awaitfordelay\x0d\x0a'00:00:10'\x0d\x0aupdatetable1\x0d\x0a\x0d\x0aset\x0d\x0aA='aa'\x0d\x0awhereB='b2'\x0d\x0acommittran\x0d\x0a\x0d\x0a同时执行,系统会检测出死锁,并中止进程\x0d\x0a\x0d\x0a补充一点:\x0d\x0aSqlServer2000支持的表级锁定提示\x0d\x0a\x0d\x0aHOLDLOCK持有共享锁,直到整个事务完成,应该在被锁对象不需要时立即释放,等于SERIALIZABLE事务隔离级别\x0d\x0a\x0d\x0aNOLOCK语句执行时不发出共享锁,允许脏读,等于READ\x0d\x0aUNCOMMITTED事务隔离级别\x0d\x0a\x0d\x0aPAGLOCK在使用一个表锁的地方用多个页锁\x0d\x0a\x0d\x0aREADPAST让sql\x0d\x0aserver跳过任何锁定行,执行事务,适用于READUNCOMMITTED事务隔离级别只跳过RID锁,不跳过页,区域和表锁\x0d\x0a\x0d\x0aROWLOCK\x0d\x0a强制使用行锁\x0d\x0a\x0d\x0aTABLOCKX强制使用独占表级锁,这个锁在事务期间阻止任何其他事务使用这个表\x0d\x0a\x0d\x0aUPLOCK\x0d\x0a强制在读表时使用更新而不用共享锁\x0d\x0a\x0d\x0a应用程序锁:\x0d\x0a应用程序锁就是客户端代码生成的锁,而不是sqlserver本身生成的锁\x0d\x0a\x0d\x0a处理应用程序锁的两个过程\x0d\x0a\x0d\x0asp_getapplock锁定应用程序资源\x0d\x0a\x0d\x0asp_releaseapplock\x0d\x0a为应用程序资源解锁\x0d\x0a\x0d\x0a注意:锁定数据库的一个表的区别\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(HOLDLOCK)\x0d\x0a其他事务可以读取表,但不能更新删除\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(TABLOCKX)\x0d\x0a其他事务不能读取表,更新和删除\x0d\x0a\x0d\x0a1\x0d\x0a如何锁一个表的某一行\x0d\x0a/*\x0d\x0a测试环境:windows2Kserver+Mssql2000\x0d\x0a\x0d\x0a所有功能都进行测试过,并有相应的结果集,如果有什么疑义在论坛跟帖\x0d\x0a\x0d\x0a关于版权的说明:部分资料来自互联网,如有不当请联系版主,版主会在第一时间处理。\x0d\x0a\x0d\x0a功能:sql遍历文件夹下的文本文件名,当然你修改部分代码后可以完成各种文件的列表。\x0d\x0a*/\x0d\x0a\x0d\x0aA\x0d\x0a连接中执行\x0d\x0a\x0d\x0aSETTRANSACTION\x0d\x0aISOLATIONLEVELREPEATABLE\x0d\x0aREAD\x0d\x0a\x0d\x0abegintran\x0d\x0a\x0d\x0aselect*fromtablename\x0d\x0awith\x0d\x0a(rowlock)whereid=3\x0d\x0a\x0d\x0awaitfordelay'00:00:05'\x0d\x0a\x0d\x0acommittran\x0d\x0a\x0d\x0aB连接中如果执行\x0d\x0a\x0d\x0aupdatetablenameset\x0d\x0acolname='10'whereid=3\x0d\x0a--则要等待5秒\x0d\x0a\x0d\x0aupdatetablename\x0d\x0aset\x0d\x0acolname='10'whereid3\x0d\x0a--可立即执行\x0d\x0a\x0d\x0a2\x0d\x0a锁定数据库的一个表\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(HOLDLOCK)\x0d\x0a\x0d\x0a注意:锁定数据库的一个表的区别\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(HOLDLOCK)\x0d\x0a\x0d\x0a其他事务可以读取表,但不能更新删除\x0d\x0a\x0d\x0aSELECT*FROMtableWITH(TABLOCKX)\x0d\x0a\x0d\x0a其他事务不能读取表,更新和删除
oracle数据库分行级锁和表级锁。用select * from table-name for update完成行级锁。用delete或update完成表级锁。你锁定的资源 别人会等待你的提交语句或回退语句完成以后再继续进行。