原文:
http://blog.csdn.net/hhb200766/article/details/6695876
1:首先,我们需要在Oracle中找到锁住的对象:
select xidusn, object_id, session_id, locked_mode from v$locked_object; |
2:得到session_id后,再根据session_id找到v$session对应的sid号和serial#:
select username,sid,serial# from v$session where sid=session_id; |
3:用oracle的slq kill掉相关的session sql语句:
alter system kill session 'sid,serial#'; |
4:如果不能kill掉,报ora-00031错:说明之前有kill过,这样找到这个进程spid slq语句:
select pro.spid from v$session ses, v$process pro where ses.sid=129 and ses.paddr=pro.addr; |
5:根据这个spid对应linux下的进程pid
linux命令:
ps -ef | grep 'ora' (找到所有ora开头的进程) kill -9 (pid) |
下面主要介绍如果这个进程是Killed状态的时候该怎么处理:
首先介绍三个视图:v$process是ORACLE处理的进程,v$session是ORACLE处理的会话,v$sqlarea是Oracle处理的SQL语句存放区就是共享池那东西。三个表联合查一下。
SELECT P.pid,S.sid,s.serial#,P.spid,S.username,S.osuser, P.serial#,P.terminal,P.program,P.background,S.status, A.sql_text FROM v$process P, v$session S,v$sqlarea A WHERE P.addr = s.paddr AND S.sql_address = a.address (+) order by P.spid; |
这里面查出来的就是Oracle实时正在处理的SQL语句过程等。当然还包含基本的(PMON)(DBW0)(LGWR)等后台进程。有时候大家在编译时,会发生怎么都编译不过去的现象,其实过程是好着的.这就是Oracle出现了死锁。死锁了肯定是锁最后一个会话,那就是你编译的那个.这时候大家有可以要重新SHUTDOWN IMMEDIATE才行解决问题.其实运行上面的语句再配合下面的就可以了。
在Oracle数据库中,可以通过kill session的方式来终止一个进程,alter system kill session 'sid,serial#' ;这里sid,serial#就是上面的v$session.sid,v$session.serial#这二值,多运行几次上面的SELECT语句就可以查出你被锁住的那个过程.其它不明的不能通杀了.这里关键就是要查出你要杀的v$session.sid,v$session.serial#二个值.然后做alter system kill session 'sid,serial#' ;做完以后.你的客户端一般就直接提示SESSION KILLED.但也有不提示没反映.再运行SELECT语句,发现v$session.status为KILLED.再等待还是一样.
从书上查知被kill掉的session,状态会被标记为killed,Oracle会在该用户下一次touch时清除该进程.Oracle就等待PMON去清除这些Session.通常等待一个被标记为Killed的Session退出需要花费很长的时间.但我从来都没等及过.这不又白杀了.没关系,用必杀招,在LINUX下杀.你用SELECT查出来的时候还有一个v$process.spid列.在LIUNX的$下运行TOP.[/img]你知发现了原来这一列的值,就是TOP下的PID值.那这样好办了.直接在LIUNX的$下 KILL -9 v$process.spid 这个值是要你在SELECT下要记录下的.好了.这就必杀了.过程编译不过去,肯定是有会话地运行.
SELECT P.pid,S.sid,s.serial#,P.spid,S.username,S.osuser, P.serial#,P.terminal,P.program,P.background,S.status,A.sql_text FROM v$process P, v$session S,v$sqlarea A WHERE P.addr = s.paddr AND S.sql_address = a.address (+) order by P.spid;和LIUNX的下的TOP你们会发现很多东西的. 比如说接口程序是很占CPU的还有查到实时会话. |
ps和top都可以看系统中正在运行的进程
ps命令提供了当前运行进程的快照。
使用带有 -ef 选项的 ps ,返回系统中所有用户的所有进程的完整列表。如果您将此 ps 命令的结果传送到 grep 中,则该结果更易于查看。例如:
$ ps -ef | grep oracle |
这条命令会显示:
UID PID PPID C STIME TTY TIME CMD oracle 1633 1 0 13:58 ?00:00:00 ora_pmon_ora1 oracle 1635 1 0 13:58 ?00:00:00 ora_dbw0_ora1 oracle 1637 1 0 13:58 ?00:00:01 ora_lgwr_ora1 oracle 1639 1 0 13:58 ?00:00:02 ora_ckpt_ora1 |
下面是ps的选项
-A:列出所有的进程。
-l:显示长列表。
-m:显示内存信息。
-w:显示加宽可以显示较多的信息。
-e:显示所有进程。
a:显示终端上的所有进程,包括其它用户的进程。
-au:显示较详细的信息。
-aux:显示所有包含其它使用者的进程。
注:如果需要即时查看最活跃的进程,可以使用"top" 。
先查看哪些表被锁住了:
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;
1.查哪个过程被锁:
查V$DB_OBJECT_CACHE视图:
SELECT * FROM V$DB_OBJECT_CACHE WHERE OWNER='过程的所属用户' AND CLOCKS!='0';
2. 查是哪一个SID,通过SID可知道是哪个SESSION:
查V$ACCESS视图:
SELECT * FROM V$ACCESS WHERE OWNER='过程的所属用户' AND NAME='刚才查到的过程名';
3. 查出SID和SERIAL#:
查V$SESSION视图:
SELECT SID,SERIAL#,PADDR FROM V$SESSION WHERE SID='刚才查到的SID';
查V$PROCESS视图:
SELECT SPID FROM V$PROCESS WHERE ADDR='刚才查到的PADDR';
4. 杀进程:
(1)先杀ORACLE进程:
ALTER SYSTEM KILL SESSION '查出的SID,查出的SERIAL#';
(2)再杀操作系统进程:
KILL -9 刚才查出的SPID或ORAKILL 刚才查出的SID 刚才查出的SPID。
Oracle的死锁
查询数据库死锁:
select t2.username||' '||t2.sid||'
'||t2.serial#||' '||t2.logon_time||'
'||t3.sql_text
from v$locked_object t1,v$session t2,v$sqltext t3
where t1.session_id=t2.sid
and t2.sql_address=t3.address
order by t2.logon_time;
查询出来的结果就是有死锁的session了,下面就是杀掉,拿到上面查询出来的SID和SERIAL#,填入到下面的语句中:
alter system kill session 'sid,serial#';
一般情况可以解决数据库存在的死锁了,或通过session id 查到对应的操作系统进程,在Unix中杀掉操作系统的进程。
SELECT a.username,c.spid AS os_process_id,c.pid
AS oracle_process_id FROM v$session a,v$process c
WHERE c.addr=a.paddr and a.sid= and a.serial#= ;
然后采用kill (unix) 或 orakill(windows )。
在Unix中:
ps -ef|grep os_process_id
kill -9 os_process_id
ps -ef|grep os_process_id
经常在Oracle的使用过程中碰到这个问题,所以也总结了一点解决方法。
1)查找死锁的进程:
sqlplus "/as sysdba" (sys/change_on_install)
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;
2)kill掉这个死锁的进程:
alter system kill session ‘sid,serial#’; (其中sid=l.session_id)
3)如果还不能解决:
select pro.spid from v$session ses,
v$process pro where ses.sid=XX and
ses.paddr=pro.addr;
其中sid用死锁的sid替换:
exit
ps -ef|grep spid
其中spid是这个进程的进程号,kill掉这个Oracle进程。
-----------------------------------------------------------------------------------------------------------------------
select v_s.machine,v_s.program,v_s.username,v_s.sid,v_s.serial# ,
pro.spid,t_o.object_name,t_o.object_type
,' kill -9 '||pro.spid
from v$session v_s,v$locked_object v_l,v$process pro,all_objects t_o where
1=1
and v_s.sid=v_l.session_id
and v_s.paddr=pro.addr
and v_s.sid=v_l.session_id
and t_o.object_id=v_l.object_id
;
-----------------------------------------------------------------------------------------------------------------------
select v_s.username,v_s.sid,v_s.serial# ,pro.spid
from v$session v_s,v$locked_object v_l,v$process pro where
1=1
and v_s.sid=v_l.session_id
and v_s.paddr=pro.addr
and v_s.sid=v_l.session_id;