Kill session 和orakill 殺會話及程式總結

bitifi發表於2015-10-09

一個使用者程式偶爾會掛起或佔用過多資源而拒絕其它會話。如果DBA依然能夠訪問資料庫,她通常可以發出以下查詢:
select s.username, s.osuser, s.sid, s.serial#, p.spid  from v$session s,v$process p
where s.paddr = p.addr    and s.username is not null;
select 'alter system kill session ',''''||trim(t2.sid)||','||trim(t2.serial#)||''';'
from v$locked_object t1,v$session t2 where t1.session_id=t2.sid order by t2.logon_time;
這個查詢將返回資料庫使用者名稱、作業系統使用者名稱、會話 ID,序列號和系統程式 ID(SPID)。然後,DBA 使用者就可以發出以下命令(前面的查詢返回的使用 SID 和SERIAL# 資訊):
ALTER SYSTEM KILL SESSION 'sid,serial#';
ALTER SYSTEM KILL SESSION '9,203';
使用這條語句有兩個問題。
第一:分配給這個程式的任何鎖或資源在會話完全超時之前不會被釋放。
第二:查詢和 kill 命令需要能夠訪問資料庫。如果一個進行失去控制,那麼資料庫訪問可能會出現問題。
在一個 UNIX 資料庫中,下一步是 ps 命令輸出的 UNIX 提示中定位程式(同樣是查詢 OSUSER 和 SPID 等 ID)然後使用 kill -9 spid 結束失控的後臺程式。然而,在 Windows 中,只有一個程式 ORACLE.EXE,而且使用者連線是在 Windows 執行緒中處理的,而不在程式中處理的。如果使用 Windows 工作管理員結束 Oracle 執行緒,就有可能影響所有使用者和後臺執行緒,並導致資料庫崩潰。
出於這些原因,Oracle 在Oracle Home/bin 目錄下提供了一個 orakill.exe 命令,這個命令的引數與ALTER SYSTEM KILL SESSION 相同,但是不要求資料庫連線。要定位一個特定的執行緒,需要尋找一個能夠顯示屬於一個程式的所有執行緒的程式。Windows 工作管理員只能顯示執行緒數和程式。你需要從微軟的資源工具包中尋找一個用於 Windows 2000 和 NT 的工具程式,比如免費的QuickSlice,或者Qslice.exe(該工具是基於 Windows 的),或者PStat(Pstat.exe 是一個命令列工具)。簡單地在 orakill 命令後輸入執行緒 ID(以十進位制表示)和 SID 即可:
orakill
orakill ORCL 2760


"Kill of thread id 2760 in instance ORCL successfully signalled[sic]."
應該只有在不能訪問資料庫來執行ALTER SYSTEM KILL SESSION 的情況才使用orakill。如果意外結束了一個必要的後臺程式,比如 PMON,那麼很可能會導致資料庫崩潰。新手永遠不要這樣做。
Orakill的使用方法如下:
              Dos提示符下:>orakill sid thread
              說明: sid Oracle的Sid號
                                  thread  Oracle的執行緒id號
在Sql*plus工具裡面可以查詢到Oracle的執行緒號
sql:>Select p.spid THREADID, s.osuser, s.program
From v$process p, v$session s
Where p.addr = s.paddr
結果如下:
THREADID  OSUSER                                  PROGRAM
--------- ----------------------- -----------------------------
169            SYSTEM                                  ORACLE.EXE
215            SYSTEM                                  ORACLE.EXE
280            SYSTEM                                  ORACLE.EXE
267            SYSTEM                                  ORACLE.EXE
287            SYSTEM                                  ORACLE.EXE
288            SYSTEM                                  ORACLE.EXE
271            SYSTEM                                  ORACLE.EXE 
282            SYSTEM                                  ORACLE.EXE 
239            PROD_NTdjones                  SVRMGRL.EXE
281            SSMITH-PCssmith              SQLPLUSW.EXE


12 rows selected.


需要注意的是,如果你Kill掉的是Oracle的核心後臺執行緒(DBWR, LGWR, SMON or PMON),將導致Oracle例項關閉。檢查Oracle的核心後臺執行緒的方法如下:
sql:>Select vb.name NOME, vp.programe PROCESSNAME, vp.spid THREADID, vs,sid SID
From v$session vs, v$process vp, v$bgprocess vb
Where vb.addr <> ‘00’ and
vb.paddr = vp.addr and
vp.addr = vs.paddr
查詢結果如下:


NOME  PROCESSNAME                                                THREADID  SID
----- ----------------------------------- --------- ------
PMON  ORACLE.EXE                                                  169            1
DBW0  ORACLE.EXE                                                  215            2
LGWR  ORACLE.EXE                                                  280            3
CKPT  ORACLE.EXE                                                  267            4
SMON  ORACLE.EXE                                                  287            5
RECO  ORACLE.EXE                                                  288            6
SNP0  ORACLE.EXE                                                  271            7
SNP1  ORACLE.EXE                                                  282            8


8 rows selected.

 

window xp + oracle 9.2.0.1


================
orakill的用法
================
SQL> SELECT spid, osuser, s.program,sid
                                FROM v$process p, v$session s
                                WHERE p.addr=s.paddr;


SPID                                        OSUSER                            PROGRAM                              SID
------------------------ -------------------- --------------- ----------
6928                                        SYSTEM                            ORACLE.EXE                            1
5272                                        SYSTEM                            ORACLE.EXE                            2
7008                                        SYSTEM                            ORACLE.EXE                            3
6588                                        SYSTEM                            ORACLE.EXE                            4
6780                                        SYSTEM                            ORACLE.EXE                            5
6128                                        SYSTEM                            ORACLE.EXE                            6
6740                                        SYSTEM                            ORACLE.EXE                            7
6684                                        SYSTEM                            ORACLE.EXE                            8
7092                                        SYSTEM                            ORACLE.EXE                            9
6272                                        SYSTEM                            ORACLE.EXE                          10
4760                                        lifeng.fang                  sqlplus.exe                                        12
SPID                                        OSUSER                            PROGRAM                              SID
------------------------ -------------------- --------------- ----------
6484                                        lifeng.fang                  sqlplus.exe                        11


4284
語法:orakill 例項名 spid  (在win上是執行緒號)


SQL> host orakill charset 6484;
Kill of thread id 6484 in instance charset successfully signalled.


spid是os程式ID
pid是Oracle process ID


根據蓋國強的猜測

當在Oracle中kill session以後, Oracle只是簡單的把相關session的paddr 指向同一個虛擬地址.此時v$process和v$session失去關聯,程式就此中斷.然後Oracle就等待PMON去清除這些Session.所以通常等待一個被標記為Killed的Session退出需要花費很長的時間.如果此時被Kill的process,重新嘗試執行任務,那麼馬上會收到程式中斷的提示,process退出,此時Oracle會立即啟動PMON來清除該session.這被作為一次異常中斷處理.

具體的實驗例子讀者可以參考

當然如果該使用者已經沒有使用,例如在決定刪除該使用者的時候,PMON很可能不回自動的去清理這些session則需要手工觸發PMON執行

首先確認PMON程式是who
SQL> select pid,spid from v$process p,v$bgprocess b
where b.paddr=p.addr
and name='PMON';
       PID SPID
---------- ------------
        2  3608
SQL> oradebug wakeup 2
已處理的語句

SQL> select p.addr from v$process p where pid <> 1  minus select s.paddr from v$session s;  查詢已經KILL的程式地址
ADDR--------
542B70E8
542B7498

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/30496894/viewspace-1813555/,如需轉載,請註明出處,否則將追究法律責任。

相關文章