文章詳情頁
Oracle中Kill session的研究
瀏覽:2日期:2023-11-17 16:05:25
Oracle中Kill session的研究 作者: Eygle link: http://www.eygle.com/faq/Kill_Session.htm 我們知道,在Oracle數據庫中,可以通過kill session的方式來終止一個進程,其基本語法結構為: alter system kill session 'sid,serial#' ; 被kill掉的session,狀態會被標記為killed,Oracle會在該用戶下一次toUCh時清除該進程. 我們發現當一個session被kill掉以后,該session的paddr被修改,假如有多個session被kill,那么多個session的paddr都被更改為相同的進程地址: SQL> select saddr,sid,serial#,paddr,username,status from v$session where username is not null;SADDRSID;SERIAL# PADDR;USERNAME;;;;STATUS-------- ---------- ---------- -------- ------------------------------ --------542E0E6C 11;;;;;314 542B70E8 EYGLE; INACTIVE542E5044 18;;;;;662 542B6D38 SYS;ACTIVESQL> alter system kill session '11,314';System altered.SQL> select saddr,sid,serial#,paddr,username,status from v$session where username is not null;SADDRSID;SERIAL# PADDR;USERNAME;;;;STATUS-------- ---------- ---------- -------- ------------------------------ --------542E0E6C 11;;;;;314 542D6BD4 EYGLE; KILLED542E5044 18;;;;;662 542B6D38 SYS;ACTIVESQL> select saddr,sid,serial#,paddr,username,status from v$session where username is not null;SADDRSID;SERIAL# PADDR;USERNAME;;;;STATUS-------- ---------- ---------- -------- ------------------------------ --------542E0E6C 11;;;;;314 542D6BD4 EYGLE; KILLED542E2AA4 14;;;;;397 542B7498 EQSPINACTIVE542E5044 18;;;;;662 542B6D38 SYS;ACTIVESQL> alter system kill session '14,397';System altered.SQL> select saddr,sid,serial#,paddr,username,status from v$session where username is not null;SADDRSID;SERIAL# PADDR;USERNAME;;;;STATUS-------- ---------- ---------- -------- ------------------------------ --------542E0E6C 11;;;;;314 542D6BD4 EYGLE; KILLED542E2AA4 14;;;;;397 542D6BD4 EQSPKILLED542E5044 18;;;;;662 542B6D38 SYS;ACTIVE 在這種情況下,很多時候,資源是無法釋放的,我們需要查詢spid,在操作系統級來kill這些進程. 但是由于此時v$session.paddr已經改變,我們無法通過v$session和v$process關聯來獲得spid 那還可以怎么辦呢? 我們來看一下下面的查詢: SQL> SELECT s.username,s.status,; 2; x.ADDR,x.KSLLAPSC,x.KSLLAPSN,x.KSLLASPO,x.KSLLID1R,x.KSLLRTYP,; 3; decode(bitand (x.ksuprflg,2),0,null,1); 4; FROM x$ksupr x,v$session s; 5; WHERE s.paddr(+)=x.addr; 6; and bitand(ksspaflg,1)! =0;USERNAME;;;;STATUSADDR;;;;KSLLAPSCKSLLAPSN KSLLASPO;;;;KSLLID1R KS D------------------------------ -------- -------- ---------- ---------- ------------ ---------- -- -;;;;;542B44A8; 0; 0;;;;0;;;;ACTIVE542B4858; 1 14 24069 0;1;;;;ACTIVE542B4C08 26 16 15901 0;1;;;;ACTIVE542B4FB8; 7 46 24083 0;1;;;;ACTIVE542B5368 12 15 24081 0;1;;;;ACTIVE542B5718 15 46 24083;; ;;;;0;1;;;;ACTIVE542B5AC8 79; 4 15923 0;1;;;;ACTIVE542B5E78 50 16 24085 0;1;;;;ACTIVE542B6228;;;;;754 15 24081 0;1;;;;ACTIVE542B65D8; 1 14 24069 0;1;;;;ACTIVE542B6988; 2 30 14571 0;1USERNAME;;;;STATUSADDR;;;;KSLLAPSCKSLLAPSN KSLLASPO;;;;KSLLID1R KS D------------------------------ -------- -------- ---------- ---------- ------------ ---------- -- -SYS;ACTIVE542B6D38; 2; 8 24071;;; ;;;0;;;;;542B70E8; 1 15 24081;;;;195 EV;;;;;542B7498; 1 15 24081;;;;195 EVSYS;INACTIVE 542B7848; 0; 0;;;;0SYS;INACTIVE 542B7BF8; 1 15 24081;;;;195 EV16 rows selected. 我們注重,紅字標出的部分就是被Kill掉的進程的進程地址. 簡化一點,其實就是如下概念: SQL> select p.addr from v$process p where pid <> 1 2 minus 3 select s.paddr from v$session s; ADDR--------542B70E8542B7498 Ok,現在我們獲得了進程地址,就可以在v$process中找到spid,然后可以使用Kill或者orakill在系統級來殺掉這些進程. 實際上,我猜測: 當在Oracle中kill session以后, Oracle只是簡單的把相關session的paddr 指向同一個虛擬地址. 此時v$process和v$session失去關聯,進程就此中斷. 然后Oracle就等待PMON去清除這些Session.所以通常等待一個被標記為Killed的Session退出需要花費很長的時間. 假如此時被Kill的process,重新嘗試執行任務,那么馬上會收到進程中斷的提示,process退出,此時Oracle會立即啟動PMON來清除該session.這被作為一次異常中斷處理. 2004年6月25日 星期五 If you have any question,please mail to [email protected] .
上一條:ORACLE回滾段管理(上)下一條:Oracle數據庫幾種關閉方式
排行榜
