Chinaunix首页 | 论坛 | 博客
  • 博客访问: 104958
  • 博文数量: 20
  • 博客积分: 64
  • 博客等级: 民兵
  • 技术积分: 100
  • 用 户 组: 普通用户
  • 注册时间: 2010-11-28 23:06
文章分类

全部博文(20)

文章存档

2019年(1)

2017年(2)

2016年(3)

2014年(10)

2013年(1)

2012年(3)

分类: Oracle

2014-03-13 15:59:46

Oracle九个重要视图
? v$lock
? v$sqlarea
? v$session
? v$sesstat
? v$session_wait
? v$process
? v$transaction
? v$sort_usage
? v$sysstat


1)v$lock
给出了锁的信息,如type字段, user type locks有3种:TM,TX,UL,system type locks有多种,常见的有:MR,RT,XR,TS等。我们只关心TM,TX锁。
当TM锁时,id1字段表示object_id;当TX锁时,trunc(id1/power(2,16))代表了回滚段号。
lmode字段,session持有的锁的模式,有6种:
  0 - none
  1 - null (NULL)
  2 - row-S (SS)
  3 - row-X (SX)
  4 - share (S)
  5 - S/Row-X (SSX)
  6 - exclusive (X)
request字段,process请求的锁的模式,取值范围与lmode相同。
ctime字段,已持有或等待锁的时间。
block字段,是否阻塞其它锁申请,当block=1时说明这个session阻塞了别的session。

2)v$sqlarea
共享池中存储的sql和一些相关的信息,如累计的执行次数(executions),逻辑读(buffer_gets),物理读(disk_reads)等统计信息。
根据address和hash_value可以定位一条sql。sql_text字段存的这个sql的前1000个字符。查找整个的sql还需去v$sqltext或者v$sqltext_with_newlines。

3)v$session
所有当前session的信息,包括一些用户信息如username,terminal,program,logon_time等,command字段表示session执行的什么命令。
通过v$session.paddr=v$process.addr跟process相关,通过v$session.taddr=v$transaction.addr跟trancation相关。
通 过sid可以到相关视图中查询与这个session相关的各种信息,如v$sesstat中有session的统计信息,v$sess_io中有 session的io统计信息,v$session_wait中有session的等待信息,v$session_event中有session在一段时 间内所等待的各种事件的信息。
根据lockwait字段可以查询当前正在等待的锁的相关信息: 
  select * from v$lock where kaddr in (select lockwait from v$session where sid=&sid);
(sql_address,sql_hash_value),(prev_sql_addr,prev_hash_value) 根据这两组字段, 可以查询到当前session正在执行的或最近一次执行的sql语句的详细信息:
  select * from v$sqltext where address = &sql_address and hash_value = &sql_hash_value;

4)v$sesstat
根据session id可以得到这个session的统计信息:
  select a.name,b.value
  from v$statname a,v$sesstat b
  where a.STATISTIC#=b.STATISTIC#
  and b.sid=&sid
  and b.value <>0
  order by b.value;

5)v$session_wait
根据session id可以得到这个session的等待情况。
event字段表示等待事件。
p1、 p2、p3告诉我们等待事件的具体含义,如果wait event是db file scattered read,p1=file_id/p2=block_id/p3=blocks,然后通过dba_extents即可确定出热点对象;如果是latch free的话,p2为闩锁号,它指向v$latch。
P1RAW,P2RAW,P3RAW列对应P1,P2,P3的十六进制值;P1TEXT,P2TEXT,P3TEXT列对应P1,P2,P3列的解释。
--求等待事件及其对应的latch
  col event format a32
  col name format a32
  select sid,event,p1 as file_id, p2 as "block_id/latch", p3 as blocks,l.name
  from v$session_wait sw,v$latch l
  where event not like '%SQL%' and event not like '%rdbms%'
  and event not like '%mon%' and sw.p2 = l.latch#(+);
--求等待事件及其热点对象
  col owner format a18
  col segment_name format a32
  col segment_type format a32
  select owner,segment_name,segment_type
  from dba_extents
  where file_id = &file_id and &block_id between block_id
  and block_id + &blocks - 1;
--综合以上两条sql,同时显示latch及热点对象(速度较慢)
  select sw.sid,event,l.name,de.segment_name
  from v$session_wait sw,v$latch l,dba_extents de
  where event not like '%SQL%' and event not like '%rdbms%'
  and event not like '%mon%' and sw.p2 = l.latch#(+) and sw.p1 = de.file_id(+) and p2 between de.block_id and de.block_id + de.blocks - 1;
--如果是非空闲等待事件,通过等待会话的sid可以求出该会话在执行的sql
  select sql_text
  from v$sqltext_with_newlines st,v$session se
  where st.address=se.sql_address and st.hash_value=se.sql_hash_value
  and se.sid =&wait_sid order by piece;
其中 STATE 列 的描述如下(参考 Oracle Database Reference 10g Release 1 Part Number B10755-01):
? 0 - WAITING (当前等待的 Session)
? -2 - WAITED UNKNOWN TIME (最后等待持续时间未知)
? -1 - WAITED SHORT TIME (最后的等待 <1/100 秒)
? >0 - WAITED KNOWN TIME (WAIT_TIME = 最后等待持续时间)

6)v$process
根据session id可以查到操作系统进程的信息:
select * from v$process where addr in (select paddr from v$session where sid=&sid );
spid字段是操作系统进程号,可以用来进行 kill -9 spid 这么一个操作。
可以找到 unix命令 top 出来的最耗cpu资源的pid,根据pid最耗资源的sql:
  SELECT
  sql_text
  FROM v$sqltext a
  WHERE (a.hash_value, a.address) IN (
  SELECT DECODE (sql_hash_value,
  0, prev_hash_value,
  sql_hash_value
  ),
  DECODE (sql_hash_value, 0, prev_sql_addr, sql_address)
  FROM v$session b
  WHERE b.paddr = (SELECT addr
  FROM v$process c
  WHERE c.spid = '&pid'))
  ORDER BY piece ASC
 /

7)v$transaction
根据session id可以查到当前session正在执行的事务信息:
select * from v$transaction where addr in (select taddr from v$session where sid=&sid );
看下面这两个字段可以看到事务进行到什么程度了:
  USED_UBLK
  NUMBER
  Number of undo blocks used

  USED_UREC
  NUMBER
  Number of undo records used
重复查询这两个值,可以看到变化,可以估计事务的进度,尤其是长时间的回滚操作,当这两个值为0,回滚也就完成了。

8)v$sort_usage
temp表空间的使用情况,当temp表空间变得巨大的时候,根据session_addr可以得到session id,根据sqladdr和sqlhash可以得到正在执行的sql:
  select se.username,se.sid,su.extents,su.blocks*to_number(rtrim(p.value)) as

  Space,tablespace,segtype,sql_text
  from v$sort_usage su,v$parameter p,v$session se,v$sql s
  where p.name='db_block_size'
  and su.session_addr=se.saddr
  and s.hash_value=su.sqlhash
  and s.address=su.sqladdr
  order by se.username,se.sid;

9)v$sysstat
所有instance的统计信息。

====================================================================
v$session and v$process说明

V$SESSION

在本视图中,每一个连接到数据库实例中的session都拥有一条记录。包括用户session及后台进程如DBWR,LGWR,arcchiver等等。

V$SESSION中的常用列

V$SESSION是基础信息视图,用于找寻用户SID或SADDR。不过,它也有一些列会动态的变化,可用于检查用户。
例如:
SQL_HASH_VALUE,SQL_ADDRESS:这两列用于鉴别默认被session执行的SQL语句。如果为null或0,那就说明这个session没有执行任何SQL语句。
PREV_HASH_VALUE和PREV_ADDRESS:两列用来鉴别被session执行的上一条语句。
(注意:当使用SQL*Plus进行选择时,确认你重定义的列宽不小于11以便看到完整的数值。)

STATUS:这列用来判断session状态是:
Achtive:正执行SQL语句(waiting for/using a resource)
Inactive:等待操作(即等待需要执行的SQL语句)
Killed:被标注为删除

下列各列提供session的信息,可被用于当一个或多个combination未知时找到session。

Session信息
SID:SESSION标识,常用于连接其它列
SERIAL#:如果某个SID又被其它的session使用的话则此数值自增加(当一个 SESSION结束,另一个SESSION开始并使用了同一个SID)。
AUDSID:审查session ID唯一性,确认它通常也用于当寻找并行查询模式
USERNAME:当前session在oracle中的用户名。

Client信息
数据库session被一个运行在数据库服务器上或从中间服务器甚至桌面通过SQL*Net连接到数据库的客户端进程启动,下列各列提供这个客户端的信息
OSUSER:客户端操作系统用户名
MACHINE:客户端执行的机器
TERMINAL:客户端运行的终端
PROCESS:客户端进程的ID
PROGRAM:客户端执行的客户端程序
要显示用户所连接PC的 TERMINAL、OSUSER,需在该PC的ORACLE.INI或Windows中设置关键字TERMINAL,USERNAME。

Application信息
调用DBMS_APPLICATION_INFO包以设置一些信息区分用户。这将显示下列各列。
CLIENT_INFO:DBMS_APPLICATION_INFO中设置
ACTION:DBMS_APPLICATION_INFO中设置
MODULE:DBMS_APPLICATION_INFO中设置
下列V$SESSION列同样可能会被用到:
 ROW_WAIT_OBJ#
 ROW_WAIT_FILE#
 ROW_WAIT_BLOCK#
 ROW_WAIT_ROW#

V$SESSION中的连接列
  Column View Joined Column(s)
  SID V$SESSION_WAIT,,V$SESSTAT,,V$LOCK,V$SESSION_EVENT,V$OPEN_CURSOR SID
  SQL_HASH_VALUE, SQL_ADDRESS V$SQLTEXT, V$SQLAREA, V$SQL HASH_VALUE, ADDRESS
  PREV_HASH_VALUE, PREV_SQL_ADDRESS V$SQLTEXT, V$SQLAREA, V$SQL HASH_VALUE, ADDRESS
  TADDR V$TRANSACTION ADDR
  PADDR V$PROCESS ADDR

示例:
1.查找你的session信息
  SELECT SID, OSUSER, USERNAME, MACHINE, PROCESS FROM V$SESSION WHERE audsid = userenv('SESSIONID');

2.当machine已知的情况下查找session
  SELECT SID, OSUSER, USERNAME, MACHINE, TERMINAL FROM V$SESSION WHERE terminal = 'pts/tl' AND machine = 'qadb27';

3.查找当前被某个指定session正在运行的sql语句。假设sessionID为100
select b.sql_text from v$session a,v$sqlarea b where a.sql_hash_value=b.hash_value and a.sid=100
寻找被指定session执行的SQL语句是一个公共需求,如果session是瓶颈的主要原因,那根据其当前在执行的语句可以查看session在做些什么。

v$process

本视图包含当前系统oracle运行的所有进程信息。常被用于将oracle或服务进程的操作系统进程ID与数据库session之间建立联系。在某些情况下非常有用:

1.如果数据库瓶颈是系统资源(如:cpu,内存),并且占用资源最多的用户总是停留在某几个服务进程,那么进行如下诸项:
1>.找出资源进程
2>.找出它们的session,你必须将进程与会话联系起来。
3>.找出为什么session占用了如此多的资源
2.SQL跟踪文件名是基于服务进程的操作系统进程ID。要找出session的跟踪文件,你必须将session与服务进程联系起来。
3.某些事件,如rdbms ipc reply,鉴别session进程的Oracle进程ID在等什么。要发现这些进程在做什么,你必须找出它们的session。
4.你所看到的服务器上的后台进程(DBWR,LGWR,PMON等)都是服务进程。要想知道他们在做什么,你必须找到他们的session。

V$PROCESS中的常用列
ADDR:进程对象地址
PID: oracle进程ID
SPID:操作系统进程ID

V$PROCESS中的连接列
 Column View Joined Column(s)
 ADDR V$SESSION PADDR

示例:
1.查找指定系统用户在oracle中的session信息及进程id,假设操作系统用户为:oracle
  select s.sid,s.SERIAL#, s.username,p.spid,s.program from v$session s, v$process p where s.osuser = 'oracle' and s.PADDR = p.ADDR

2.查看锁和等待
  SELECT
  lpad(' ', decode(l.xidusn, 0, 3, 0)) || l.oracle_username User_name,
  o.owner,o.object_name,o.object_type,s.sid,s.serial#,p.spid
  FROM v$locked_object l, dba_objects o, v$session s, v$process p
  WHERE l.object_id = o.object_id
  AND l.session_id = s.sid and s.paddr = p.addr
  ORDER BY o.object_id, xidusn DESC

附注:
  在linux环境可以通过ps查看进程信息包括pid,windows中任务管理器的PID与v$process中pid不能一一对应,这块在 oracleDocument中也没有找到介绍,后来google了一下,有资料介绍说是由于windows是多线程服务器,每个进程包含一系列线程。这 点于unix等不同,Unix每个Oralce进程独立存在,在Nt上所有线程由Oralce进程衍生。
  要在windows中显示oracle相关进程pid,我们可以通过一个简单的sql语句来实现。
SELECT s.SID, p.pid, p.spid signaled, s.osuser, s.program
FROM v$process p, v$session s
WHERE p.addr = s.paddr;

Eygle大师写了一段sql脚本getsql.sql,用来获取指定pid正在执行的sql语句,在此也附注上来。
REM getsql.sql
REM author eygle
REM 在windows上,已知进程ID,得到当前正在执行的语句
REM 在windows上,进程ID为16进制,需要转换,在UNIX直接为10进制
  SELECT
  sql_text
  FROM v$sqltext a
  WHERE (a.hash_value, a.address) IN (
  SELECT DECODE (sql_hash_value,
  0, prev_hash_value,
  sql_hash_value
  ),
  DECODE (sql_hash_value, 0, prev_sql_addr, sql_address)
  FROM v$session b
  WHERE b.paddr = (SELECT addr
  FROM v$process c
  WHERE c.spid = TO_NUMBER ('&pid', 'xxxx')))
  ORDER BY piece ASC
  /

阅读(2663) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~