JAVA中锁概及运用.docx
- 文档编号:28136483
- 上传时间:2023-07-08
- 格式:DOCX
- 页数:25
- 大小:26.24KB
JAVA中锁概及运用.docx
《JAVA中锁概及运用.docx》由会员分享,可在线阅读,更多相关《JAVA中锁概及运用.docx(25页珍藏版)》请在冰豆网上搜索。
JAVA中锁概及运用
一、锁的概念及分类
ORACLE数据库是现今数据库领域应用最广泛的,同时它也是一个庞大的系统,全面了解它、玩转它不但需要一定的理论知识,更需要开发经验与工程经验。
本人是ORACLE一爱好者,以下是本人对ORACLE锁的一些经验,希望能与大家共同分享。
ORACLE锁具体分为以下几类:
1.按用户与系统划分,可以分为自动锁与显示锁
自动锁:
当进行一项数据库操作时,缺省情况下,系统自动为此数据库操作获得所有有必要的锁。
显示锁:
某些情况下,需要用户显示的锁定数据库操作要用到的数据,才能使数据库操作执行得更好,显示锁是用户为数据库对象设定的。
2.按锁级别划分,可分为共享锁、共享更新锁、排它锁
共享锁:
共享锁使一个事务对特定数据库资源进行共享访问——另一事务也可对此资源进行访问或获得相同共享锁。
共享锁为事务提供高并发性,但如拙劣的事务设计+共享锁容易造成死锁或数据更新丢失。
―锁定表
―仅允许其他用户执行查询操作
―不能插入、更新和删除
―多个用户可以同时在同一表中放置此锁
―locktabletable_name
―insharemode[nowait];
―rollback和commit命令释放锁
―nowait关键字告诉其他用户不用等
共享更新锁
―锁定要被更新的行
―允许其他用户同时查询、插入、更新未被锁定的行
―在SELECT语句中使用“FORUPDATE”子句,可以强制使用共享更新锁
―允许多个用户同时锁定表的不同行
加锁的两种方法
1locktabletab_nameinshareupdatemode;
2selectcolumn1,column2
fromgoods
wheregoods
wheregid=1001
forupdateofcolumn1,column2
排它锁:
事务设置排它锁后,该事务单独获得此资源,另一事务不能在此事务提交之前获得相同对象的共享锁或排它锁。
―与其他两种锁相比,排他锁是限制性最强的表锁
―仅允许其他用户查询数据
―不允许执行插入、删除和更新操作
―在同一时间仅允许一位用户在表上放置排他锁
―共享锁与此相反
locktabletab_nameinexclusivemode;
locktable<表名>[<表名>]...
insharemode[nowait]
locktable<表名>[<表名>]...
inexclusivemode[nowait]
locktable<表名>[<表名>]...
inshareupdatemode[nowait]
3.按操作划分,可分为DML锁、DDL锁
+DML锁又可以分为,行锁、表锁、死锁
-行锁:
当事务执行数据库插入、更新、删除操作时,该事务自动获得操作表中操作行的排它锁。
手动锁定方式:
SELECT…FORUPDATE子句
―在表的一行或多行上放置排他锁
―用于防止其他用户更新该行
―可以执行除更新之外的其他操作
―select*fromgoodswheregid=1001
―forupdateofgname;
―只有该用户提交事务,其他用户才能够更新gname
FORUPDATEWAIT子句
―Oracle9i中的新增功能
―防止无限期地等待锁定的行
―等待间隔必须指定为数值文字
―等待间隔不能是表达式、赋值变量或PL/SQL
变量
―select*fromgoodswheregid=1001forupdateofgnamewait3
―等待用户释放更新锁的时间为3秒,否则超时。
-表级锁:
当事务获得行锁后,此事务也将自动获得该行的表锁(共享锁),以防止其它事务进行DDL语句影响记录行的更新。
事务也可以在进行过程中获得共享锁或排它锁,只有当事务显示使用LOCKTABLE语句显示的定义一个排它锁时,事务才会获得表上的排它锁,也可使用LOCKTABLE显示的定义一个表级的共享锁(LOCKTABLE具体用法请参考相关文档)。
―可以设置为三种模式:
共享、共享更新和排他
语法:
locktable
-死锁:
当两个事务需要一组有冲突的锁,而不能将事务继续下去的话,就出现死锁。
如事务1在表A行记录#3中有一排它锁,并等待事务2在表A中记录#4中排它锁的释放,而事务2在表A记录行#4中有一排它锁,并等待事务;1在表A中记录#3中排它锁的释放,事务1与事务2彼此等待,因此就造成了死锁。
死锁一般是因拙劣的事务设计而产生。
死锁只能使用SQL下:
altersystemkillsession"sid,serial#";或者使用相关操作系统kill进程的命令,如UNIX下kill-9sid,或者使用其它工具杀掉死锁进程。
+DDL锁又可以分为:
排它DDL锁、共享DDL锁、分析锁
-排它DDL锁:
创建、修改、删除一个数据库对象的DDL语句获得操作对象的排它锁。
如使用altertable语句时,为了维护数据的完成性、一致性、合法性,该事务获得一排它DDL锁。
-共享DDL锁:
需在数据库对象之间建立相互依赖关系的DDL语句通常需共享获得DDL锁。
如创建一个包,该包中的过程与函数引用了不同的数据库表,当编译此包时,该事务就获得了引用表的共享DDL锁。
-分析锁:
ORACLE使用共享池存储分析与优化过的SQL语句及PL/SQL程序,使运行相同语句的应用速度更快。
一个在共享池中缓存的对象获得它所引用数据库对象的分析锁。
分析锁是一种独特的DDL锁类型,ORACLE使用它追踪共享池对象及它所引用数据库对象之间的依赖关系。
当一个事务修改或删除了共享池持有分析锁的数据库对象时,ORACLE使共享池中的对象作废,下次在引用这条SQL/PLSQL语句时,ORACLE重新分析编译此语句。
4.内部闩锁
内部闩锁:
这是ORACLE中的一种特殊锁,用于顺序访问内部系统结构。
当事务需向缓冲区写入信息时,为了使用此块内存区域,ORACLE首先必须取得这块内存区域的闩锁,才能向此块内存写入信息。
Oracle锁表
行级锁
表级锁
行级锁
----行被排他锁定
----在某行的锁被释放之前,其他用户不能修改此行
----使用commit或rollback命令释放锁
----Oracle通过使用INSERT、UPDATE和SELECT…FORUPDATE语句自动获取行级锁
SELECT…FORUPDATE子句
―在表的一行或多行上放置排他锁
―用于防止其他用户更新该行
―可以执行除更新之外的其他操作
―select*fromgoodswheregid=1001
―forupdateofgname;
―只有该用户提交事务,其他用户才能够更新gname
FORUPDATEWAIT子句
―Oracle9i中的新增功能
―防止无限期地等待锁定的行
―等待间隔必须指定为数值文字
―等待间隔不能是表达式、赋值变量或PL/SQL
变量
―select*fromgoodswheregid=1001forupdateofgnamewait 3
―等待用户释放更新锁的时间为3秒,否则超时。
•表级锁
―保护表的数据
―在多个用户同时访问数据时确保数据的完整性
―可以设置为三种模式:
共享、共享更新和排他
语法:
locktable
共享锁
―锁定表
―仅允许其他用户执行查询操作
―不能插入、更新和删除
―多个用户可以同时在同一表中放置此锁
―locktabletable_name
―insharemode[nowait];
―rollback和commit命令释放锁
―nowait关键字告诉其他用户不用等待
共享更新锁
―锁定要被更新的行
―允许其他用户同时查询、插入、更新未被锁定的行
―在SELECT语句中使用“FORUPDATE”子句,可以强制使用共享更新锁
―允许多个用户同时锁定表的不同行
加锁的两种方法
locktabletab_nameinshareupdatemode;
selectcolumn1,column2
fromgoods
wheregoods
wheregid=1001
forupdateofcolumn1,column2
排他锁
―与其他两种锁相比,排他锁是限制性最强的表锁
―仅允许其他用户查询数据
―不允许执行插入、删除和更新操作
―在同一时间仅允许一位用户在表上放置排他锁
―共享锁与此相反
locktabletab_nameinexclusivemode;
locktable<表名>[<表名>]...
insharemode[nowait]
locktable<表名>[<表名>]...
inexclusivemode[nowait]
locktable<表名>[<表名>]...
inshareupdatemode[nowait]
二、oracle中如何区分行级锁,表级锁、死锁以及如何处理。
在ORACLE中,为了保证数据的一致性,在对数据库中的数据进行操作时,系统会进行对数据相应的锁定。
当程序对所做的修改进行提交(commit)或回滚后(rollback)后,锁住的资源便会得到释放,从而允许其它用户进行操作。
但是,有时,由于程序中的原因,锁住资源后长时间未对其工作进行提交;或是由于用户的原因,如调出需要修改的数据后,未及时修改并提交,而是放置于一旁;或是由于客户服务器方式中客户端出现"死机",而服务器端却并未检测到,从而造成锁定的资源未被及时释放,影响到其它用户的操作。
这时,我们需要迅速地诊断出锁住资源的用户并解决其锁定。
1.诊断系统中的锁
为了找出系统中那些用户锁住资源以及那些用户在等待相应的资源,可使用以下语句(其中的/*+NO_MERGE(..)*/千万不可省略,否则会很慢):
--looklock.sql
--usetheNO_MERGEhintscanspeedupthequery
select/*+NO_MERGE(a)NO_MERGE(b)NO_MERGE(c)*/'Wait'"Status",a.username,a.machine,a.sid,a.serial#,a.last_call_et"Seconds",b.id1,c.sql_text"SQL"
fromv$sessiona,v$lockb,v$sqltextc
wherea.usernameisnotnull
anda.lockwait=b.kaddr
andc.hash_value=a.sql_hash_value
union
select/*+NO_MERGE(a)NO_MERGE(b)NO_MERGE(c)*/'Lock'"Status",a.username,a.machine,a.sid,a.serial#,a.last_call_et"Seconds",b.id1,c.sql_text"SQL"
fromv$sessiona,v$lockb,v$sqltextc
whereb.id1in
(select/*+NO_MERGE(d)NO_MERGE(e)*/distincte.id1
fromv$sessiond,v$locke
whered.lockwait=e.kaddr)
anda.usernameisnotnull
anda.sid=b.sid
andb.request=0
andc.hash_value=a.sql_hash_value;
执行后的结果如下所示:
StatUSERNAMEMACHINESIDSERIAL#SecondsID1
--------------------------------------------------------------------------------------
SQL
----------------------------------------------------------------
LockCIQUSRCIQ\DULMACER12966245131089
select*fromc_trade_modeforupdate
WaitCIQUSRCIQ\DULMACER10735111131089
updatec_trade_modesetx_name='zzz'wherex_code='5'
WaitCIQUSRCIQ\DULMACER151061094131089
select*fromc_trade_modeforupdate
其中:
Status有两种状态,LOCK表明该进程锁住了某个资源,WAIT表示该进程正在等待某个资源。
Username,Machine分别为ORACLE用户名及机器名
SID,SERIAL#可用于随后的解锁操作
Seconds表示该进程最后一次进行操作至当前的时间(秒)
ID1,锁标识。
某个LOCK状态的ID1与某个WAIT状态的ID1相同,可说明锁的正是另一个进程等待的。
SQL:
锁住资源的SQL语句
2.解除锁
诊断出锁的状态后,若发现该阻塞其它用户进程的进程是正常操作中,则可通知该用户对其进行提交,从而达到释放锁资源的目的;若为非正常操作,即,其状态为"inactive",且其Seconds已为较多长时间,则可执行以下语句将该进程进行清除,系统会自动对其进行回滚,从而释放锁住的资源。
altersystemkillsession'sid,serial#';
例如:
对于上例中显示的结果,可用以下语句清除锁住资源的进程:
altersystemkillsession'12,966';
在网络断掉(通过拔掉网线)或非正常终止进程(通过taskmanager强行关闭sql*plus)时,oracle在有限的时间内(我只观查了5-10分)内,oracle未能对该进程作任何处理。
这个处理与TCP协议有关,因为SQLNET在使用TCP/IP协议进行网络连接时是一种短连接,当ORACLE连接异常终止时,因为是异常终止,终止信号并没有通过网络通知server端,因此只有下次server有结果从服务器端返回需与client通信时,server才会发现此client已经端掉。
因此出现你前面所提ORACLE处理异常终止进程延时情况.
死锁:
你可以试验一条彼此存在依赖关系的update语句,ORACLE处理这种锁时不是很好。
查锁语句:
查询产生锁的用户锁sql
selecta.usernameusername,a.sidsid,a.serial#serial,b.id1id1,c.sql_textsqltext
fromv$sessiona,v$lockb,v$sqltextc
whereb.id1in
(selectdistincte.id1
fromv$sessiond,v$locke
whered.lockwait=e.kaddr)
anda.sid=b.sid
andc.hash_value=a.sql_hash_value
andb.request=0;
关于死锁的诊断:
记录死锁的视图主要有两个v$lock,v$locked_object
查询相关的锁时可以关联下表ALL_OBJECTS取出其中的表对象,
selecta.SID,b.object_namefromv$locka,all_objectsbwherea.id1=b.object_idanda.block=1;
selectsid,serial#,last_call_et,status,program,usernamefromv$sessionwheresidin(selectsidfromv$lockwhereblock=1)
然后根据SID在V$SESSION视图里KILL掉就OK了
altersystemkillsession'sid,serial#';
三
查找造成相应锁的sql语句
1.FindingtheSessionsHoldingtheLock
Findthe(ID1,ID2,type)forsessionswaitingforalock(LMODE=0).
Findthesessionholdingthelock(REQUEST=0)forthatID1,ID2,type.
SELECTlpad('',DECODE(request,0,0,1))||sidsess,id1,id2,lmode,request,type
FROMV$LOCK
WHEREid1IN(SELECTid1FROMV$LOCKWHERElmode=0)
ORDERBYid1,request
SIDID1ID2LMODEREQUESTTY
------------------------------------------------
123719670520049360TX<-LockHolder
125619670520049306TX<-LockWaiter
117619670520049306TX<-LockWaiter
93858985420135260TX<-LockHolder
163458985420135206TX<-LockWaiter
2.FindingtheStatementsbeingExecutedbyTheseSessions
SELECTsid,sql_address,sql_hash_value
FROMV$SESSION
WHERESIDIN(1237,1256,1176,938,1634);
SIDSQL_HASH_VALUE
-------------------
9382078523611<-Holder
11761646972797<-Waiter
12373735785744<-Holder
12561141994875<-Waiter
16342417993520<-Waiter
3.FindingtheTextforTheseSQLStatements
SELECT*FROMv$sqltextt
WHEREt.ADDRESS=''--代入上一步查到的sql_address
ANDt.HASH_VALUE=''--代入上一步查到的sql_hash_value
ORDERBYt.ADDRESS,t.HASH_VALUE,t.COMMAND_TYPE,t.PIECE;
4.还可查以下两个表:
select*fromdba_waiters/dba_blockers
四、一般锁的管理
今天,做了一个oracle数据库的存储过程,可当中一个游标将表给锁住了。
弄了半天,一开始用kill3444直接将进程杀死。
可是命令对它没有一点作用,数据库锁一直呆在那里。
后来在google一搜,发现一篇很好文章。
摘录在后面。
即刻杀死了数据库锁。
方法如下:
1、查看数据库锁,诊断锁的来源及类型:
selectobject_id,session_id,locked_modefromv$locked_object;
或者用以下命令:
selectb.owner,b.object_name,l.session_id,l.locked_mode
fromv$locked_objectl,dba_objectsb
whereb.object_id=l.object_id
2、找出数据库的serial#,以备杀死:
selectt2.username,t2.sid,t2.serial#,t2.logon_time
fromv$locked_objectt1,v$sessiont2
wheret1.session_id=t2.sidorderbyt2.
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- JAVA 中锁概 运用