在数据库的开发过程中,经常碰到包、存储过程、函数无法编译或编译时会导致PL/SQL 无法响应的问题。碰到这种问题,基本上都要重启数据库解决,严重浪费开发时间。本文将就产生这种现象的原因和解决方案做基本的介绍。

解决方案

碰到这种问题,如果知道是被谁锁定了(可以查出来的),可以让对方尽快把锁释放掉;实在查不出来只能手工将这个锁杀掉了。步骤如下:

1、     首先查出哪些进程锁住了这个对象,语句如下:

Selectb.SID,b.SERIAL#

Fromdba_ddl_locks a, v$session b

Where a.session_id= b.SID

And a.name =’FUN_CORE_SERVICECALL’;

2、     执行如下语句杀进程:alter system kill session ‘sid,serial#’

3、     执行了以上的语句后,有的时候不一定能够将进程杀掉。这个时候就需要连到数据库服务器上杀掉服务器端的进程了,查询语句:

Select spid, osuser, s.program

Fromv$session s, v$process p

Where s.paddr= p.addr

And s.sid =(上面查出来的SID)

4、     在服务器上执行如下语句:

#kill -9 spid(UNIX平台)

orakill sid thread(Windows平台SID是Oracle的实例名,thread是上面查出来的SID)

5、     执行完4步以后基本上就可以杀掉这些死锁的进程了,不放心的话可以再执行第一步确认下。

本文只能是说对这个问题做了初步分析,产生此问题的原因及解决方案涉及到很多Oracle的基本知识。如:DML语句,DDL语句;Oracle锁的机制;v$session 与 v$process之间关系等。有兴趣的兄弟可以更加深入的研究,欢迎跟我交流!

死锁是数据库经常发生的问题,数据库一般不会无缘无故产生死锁,死锁通常都是由于我们应用程序的设计本身造成的。产生死锁时,如何解决呢,下面是常规的解决办法:

1)执行下面SQL,先查看哪些表被锁住了:

select b.owner,b.object_name,a.session_id,a.locked_mode

from v$locked_object a,dba_objects b

where b.object_id = a.object_id;

2)查处引起死锁的会话

select b.username,b.sid,b.serial#,logon_time

from v$locked_object a,v$session b

where a.session_id = b.sid order by b.logon_time;

这里会列出SID

3) 查出SID和SERIAL#:

查V$SESSION视图:

SELECT SID,SERIAL#,PADDR FROM V$SESSION WHERE SID=’刚才查到的SID’;

这一步将得到PADDR

4)查V$PROCESS视图:

SELECT SPID FROM V$PROCESS WHERE ADDR=’刚才查到的PADDR’;

这一步得到SPID

5)杀死进程

(1)在数据库中,杀掉ORACLE进程:

ALTER SYSTEM KILL SESSION ‘查出的SID,查出的SERIAL#’;

执行存储过程更新一个表中的数据的时候产生如下的错误:

SQL> exec update_jc_kxx_yxrq;

begin update_jc_kxx_yxrq; end;

ORA-20998: Err=-2049,Msg=0-ORA-02049: 超时: 分布式事务处理等待锁定

ORA-06512: 在”ICUSER.UPDATE_JC_KXX_YXRQ”, line 36

ORA-06512: 在line 2

以sys用户登陆数据库查询死锁

SQL> select username,lockwait,status,machine,program from v$session where sid in

(select session_id from v$locked_object);

USERNAME LOCKWAIT STATUS

—————————— —————- ——–

MACHINE

—————————————————————-

PROGRAM

————————————————

icdb

JDBC Thin Client

ICUSER 000000038A37C0C8 ACTIVE

icdb

JDBC Thin Client

说明数据库有死锁

然后使用一下语句查找被死锁的语句

SQL> select sql_text from v$sql where hash_value in

(select sql_hash_value from v$session where sid in

(select session_id from v$locked_object));

SQL_TEXT

——————————————————————————–

update JC_KXX SET LJXF =NVL ( LJXF , 0 ) + :1 , YE =:2 WHERE KH =:3

update jc_kxx set zt=’07’ where kh=’1000530330′

再使用以下语句查找被死锁的进程

SQL> SELECT s.username,l.OBJECT_ID,l.SESSION_ID,s.SERIAL#,

l.ORACLE_USERNAME,l.OS_USER_NAME,l.PROCESS

FROM V$LOCKED_OBJECT l,V$SESSION S WHERE l.SESSION_ID=S.SID;

USERNAME OBJECT_ID SESSION_ID SERIAL#

—————————— ———- ———- ———-

ORACLE_USERNAME OS_USER_NAME PROCESS

—————————— —————————— ————

ICUSER 30523 32 42463

ICUSER oracle

ICUSER 30523 28 25508

ICUSER oracle

ICUSER 30523 76 14781

ICUSER oracle

USERNAME OBJECT_ID SESSION_ID SERIAL#

—————————— ———- ———- ———-

ORACLE_USERNAME OS_USER_NAME PROCESS

—————————— —————————— ————

ICUSER 30523 24 37522

ICUSER oracle

使用一下语句把死锁的进程kill

alter system kill session ‘sid,serial#’; (其中sid=l.session_id)

如:SQL> alter system kill session ‘24,37522’;

再次执行存储过程,错误没有了。语句执行成功!

 

 

发表回复

您的电子邮箱地址不会被公开。