`
thinktothings
  • 浏览: 766942 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

教你快速掌握Oracle中"Kill"进程的方法 .

 
阅读更多

教你快速掌握Oracle中"Kill"进程的方法

 

原文:

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;

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics