继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

拜年+散分贴《Oracle SQL_TRACE和10046事件优化SQL实例》

PIPIONE
关注TA
已关注
手记 1063
粉丝 147
获赞 702


一 数据库版本

LEO1@LEO1>select * from v$version;

BANNER

--------------------------------------------------------------------------------

Oracle Database11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production

PL/SQL Release11.2.0.1.0 - Production

CORE    11.2.0.1.0      Production

TNS for Linux:Version 11.2.0.1.0 - Production

NLSRTL Version11.2.0.1.0 - Production

 

二 演示使用SQL_TRACE和10046事件对其它会话进行跟踪,并给出trace结果

SQL_TRACE:Oracle这个功能主要是为了追踪SQL的执行过程,分析SQL的性能,资源消耗情况。

1.查看SQL是如何操作处理数据

2.查看SQL在执行过程中产生了的等待事件

3.查看SQL的执行过程资源消耗

4.查看SQL的实际执行计划

5.查看SQL的递归语句

6.如果要探索SQL如何执行的可以详细看看

10046:用于分析SQL执行过程中性能消耗情况,可以查看绑定变量信息,可以查看等待事件信息,它比SQL_TRACE输入输出更多参数。

 

上述工具使用场合:1.优化SQL语句

                  2.查看SQL语句执行计划

                  3.跟踪SQL语句执行过程

                  4.把会话中SQL的信息重定向到一个文件里

 

SET AUTO TRACE:1.输出SQL语句估算的执行计划(猜出来的)

                2.SQL语句并没有真正执行,只关注这条SQL的执行计划对不对

                3.只是用来估算执行计划

 

实验

使用SQL_TRACE对其它会话进行跟踪

如果对当前会话进行跟踪只需alter session set sql_trace=true;即可,如果对其它会话进行跟踪还需要设置另外一些参数。

我们现在做一下,从144会话跟踪12会话的SQL

144会话我们使用leo1用户操作

12会话我们使用leo2用户操作

144会话

LEO1@LEO1> selectdistinct sid from v$mystat;    可以查询当前会话ID

       SID

----------------

       144

我们用会话ID和串号来唯一定位一个会话,现在我们把2个会话信息都显示出来了

LEO1@LEO1>select sid,serial# from v$session where sid in (144,12);

       SID   SERIAL#

---------------------------------

      12      4472

     144       979

这时我有了一个疑问,定位一个会话一般来说看sid就可以了,那么为什么后面还跟着个serial呢,这个serial是干什么用的呢,咨询了一下Alantany 查了一下官方文档

SID NUMBER:Sessionidentifier   就是会话标识

SERIAL#  NUMBER  :是用来标识唯一一个会话操作对象的,保证这个会话发出的命令可以正确的应用到对应的会话对象上。

场合 一个会话的结束和另一个会话开始都使用了同一个SID,区分这是2个不同的会话

例子

第一次leonarding登陆sid=12,操作了leo1表,退出

  SID   SERIAL#

---------------------------------

  12       4472

第二次Alan登陆sid=12,又操作了leo2表,退出

  SID   SERIAL#

---------------------------- ----

  12      4777

如果只是看SID我们不能分辨出是谁登录了会话操作了leo1表和leo2表,而serial可以分辨出不同会话的命令正确应用到对应的对象上,区分这是2个不同的人登录的会话。

 

LEO1@LEO1> droptable leo1;                  清理环境

Table dropped.

LEO1@LEO1>create table leo1 as select * from dba_objects;    用leo1用户创建leo1表

Table created.

LEO1@LEO1>select count(*) from leo1;                    看看有多少条记录

  COUNT(*)

----------------

     72007

LEO1@LEO1>execute dbms_stats.gather_table_stats('LEO1','LEO1',method_opt=>'for allcolumns size 254');

PL/SQL proceduresuccessfully completed.

随便做个表分析和直方图

LEO1@LEO1> conn/ as sysdba                  切换为管理员

Connected.

SYS@LEO1> grantexecute on dbms_system to leo1; 授予执行“系统包”的用户权限给leo1,必须授予否则报错

ERROR atline 1:

ORA-06550:line 1, column 7:

PLS-00201:identifier 'DBMS_SYSTEM.SET_SQL_TRACE_IN_SESSION' must be declared

ORA-06550:line 1, column 7:

PL/SQL:Statement ignored

SYS@LEO1> connleo1/leo1                     我们在切换回来

LEO1@LEO1>execute sys.dbms_system.set_sql_trace_in_session(12,4472,true);

PL/SQL proceduresuccessfully completed.

启动跟踪会话ID=12,SERIAL=4472的SQL

声明:这个存储过程是SYS用户特有的,所以在引用时必须带schema,不带就会报错如下

ERROR atline 1:

ORA-06550:line 1, column 7:

PLS-00201:identifier 'SYS.DBMS_SYSTEM' must be declared

ORA-06550:line 1, column 7:

PL/SQL:Statement ignored

12会话

LEO2@LEO1> select /*+ trace_by_leo1_session*/ count(*) from leo1.leo1;    leo2用户查询leo1表

  COUNT(*)

----------------

     72007                                          这是执行了第一条语句

/*+trace_by_leo1_session */  用来标识这条SQL语句是leo2用户发出的,查询的是leo1用户的表

select /*+trace_by_leo1_session */ count(*) from leo1.leo1   这是执行了第二条语句

select /*+trace_by_leo1_session */ object_type,count(*) from leo1.leo1 group byobject_type   

                                                    这是执行了第三条语句

我们从trace文件里面找一找追踪到这三条语句了嘛

144会话

LEO1@LEO1>execute sys.dbms_system.set_sql_trace_in_session(12,4472,false);

PL/SQL proceduresuccessfully completed.

关闭跟踪会话ID=12,SERIAL=4472的SQL(exit sqlplus也可以达到同样效果)

LEO1@LEO1>select * from v$diag_info where name like 'Default Trace File';

INST_ID   NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

1        Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_8420.trc

指定当前会话写入的trace文件,这里面就记载了我刚才追踪的12会话上执行的SQL信息

如果你用的是Oracle 10g则目录不同请注意

[oracle@leonarding1~]$ cd /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/

[oracle@leonarding1trace]$ ll | grep LEO1_ora_8420.trc

显示没有这个文件,退出试试

LEO1@LEO1> exit

Disconnected fromOracle Database 11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production

With thePartitioning, OLAP, Data Mining and Real Application Testing options

还是没有,后来我根据执行时间猜出是LEO1_ora_7455.trc这个文件,不知道是怎么回事

[oracle@leonarding1trace]$ vim LEO1_ora_7455.trc    我们阅读一下原始trace文件

=======================================================================================

PARSING IN CURSOR#4 len=59 dep=0 uid=86 oct=3 lid=86 tim=1360332670938439 hv=3157118559ad='7d950658' sqlid='42nypzay2vmkz'

select/*+ trace_by_leo1_session */ count(*) from leo1.leo1      第一条语句,有1次硬解析

END OF STMT

PARSE#4:c=5000,e=28805,p=0,cr=6,cu=0,mis=1,r=0,dep=0,og=1,plh=4191104944,tim=1360332670938434

EXEC#4:c=0,e=97,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=4191104944,tim=1360332670938668

FETCH#4:c=20997,e=21272,p=0,cr=1031,cu=0,mis=0,r=1,dep=0,og=1,plh=4191104944,tim=1360332670960096

STAT #4 id=1 cnt=1pid=0 pos=1 obj=0 op='SORT AGGREGATE (cr=1031 pr=0 pw=0 time=0 us)'

STAT #4 id=2cnt=72007 pid=1 pos=1 obj=74145 op='TABLE ACCESS FULL LEO1 (cr=1031 pr=0 pw=0 time=623242us cost=288 size=0 card=72007)'

FETCH#4:c=0,e=11,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=0,plh=4191104944,tim=1360332670960834

*** 2013-02-0822:12:08.997

CLOSE#4:c=0,e=44,dep=0,type=0,tim=1360332728997966

使用的是4号游标,PARSE #4 解析 -> EXEC #4 执行 ->  FETCH #4  取操作 -> CLOSE #4 关闭游标

=======================================================================================

PARSING IN CURSOR#2 len=59 dep=0 uid=86 oct=3 lid=86 tim=1360330870718759 hv=3157118559 ad='7d950658'sqlid='42nypzay2vmkz'

select/*+ trace_by_leo1_session */ count(*) from leo1.leo1      第二条语句,还有1次硬解析

END OF STMT

PARSE#2:c=3000,e=2516,p=0,cr=0,cu=0,mis=1,r=0,dep=0,og=1,plh=4191104944,tim=1360330870718752

EXEC#2:c=0,e=62,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=4191104944,tim=1360330870719008

FETCH#2:c=19997,e=20021,p=0,cr=1031,cu=0,mis=0,r=1,dep=0,og=1,plh=4191104944,tim=1360330870739148

STAT #2 id=1 cnt=1pid=0 pos=1 obj=0 op='SORT AGGREGATE (cr=1031 pr=0 pw=0 time=0 us)'

STAT #2 id=2cnt=72007 pid=1 pos=1 obj=74145 op='TABLE ACCESS FULL LEO1 (cr=1031 pr=0 pw=0time=477183 us cost=288 size=0 card=72007)'

FETCH#2:c=0,e=6,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=0,plh=4191104944,tim=1360330870739918

CLOSE#7:c=0,e=37,dep=0,type=0,tim=1360332670909360

使用的是2号游标,PARSE #2 解析 -> EXEC #2 执行 ->  FETCH #2  取操作

=======================================================================================

PARSING IN CURSOR#3 len=92 dep=0 uid=86 oct=3 lid=86 tim=1360332728999761 hv=1179742142ad='7f52a418' sqlid='afbr2j5352vxy'

select/*+ trace_by_leo1_session */ object_type,count(*) from leo1.leo1 group byobject_type

                                                        第三条语句

END OF STMT

PARSE#3:c=1000,e=1664,p=0,cr=0,cu=0,mis=1,r=0,dep=0,og=1,plh=908401892,tim=1360332728999755

EXEC#3:c=0,e=102,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=908401892,tim=1360332728999981

FETCH#3:c=38994,e=38233,p=0,cr=1031,cu=0,mis=0,r=1,dep=0,og=1,plh=908401892,tim=1360332729038339

FETCH#3:c=0,e=203,p=0,cr=0,cu=0,mis=0,r=15,dep=0,og=1,plh=908401892,tim=1360332729039339

FETCH#3:c=0,e=177,p=0,cr=0,cu=0,mis=0,r=15,dep=0,og=1,plh=908401892,tim=1360332729040527

FETCH#3:c=0,e=236,p=0,cr=0,cu=0,mis=0,r=12,dep=0,og=1,plh=908401892,tim=1360332729041957

STAT #3 id=1cnt=43 pid=0 pos=1 obj=0 op='HASH GROUP BY (cr=1031 pr=0 pw=0 time=0 uscost=290 size=387 card=43)'

STAT #3 id=2cnt=72007 pid=1 pos=1 obj=74145 op='TABLE ACCESS FULL LEO1 (cr=1031 pr=0 pw=0time=497135 us cost=288 size=648063 card=72007)'

*** 2013-02-0822:12:21.604

CLOSE #3:c=0,e=34,dep=0,type=0,tim=1360332741604218

使用的是3号游标,PARSE #3 解析 -> EXEC #3 执行 ->  FETCH #3  取操作 -> CLOSE #3 关闭游标

每次执行都使用了不同的游标号,由此可见我们顺利的跟踪了12会话SQL

=======================================================================================

这个原始trace文件内容比较多,不是很好看,需要花很多时间来斟酌,如果想详细了解SQL整个执行过程的话可以慢慢研究一下。下面我们来看用tkprof工具汇总的精简版report

[oracle@leonarding1trace]$ tkprof LEO1_ora_7455.trc sql_trace.txt sys=no

TKPROF: Release11.2.0.1.0 - Development on Fri Feb 8 22:50:10 2013

Copyright (c)1982, 2009, Oracle and/or its affiliates. All rights reserved.

sys=no 参数表示过滤掉sys用户生成的递归语句,这样可以精简出比较好看的报告

[oracle@leonarding1trace]$ vim sql_trace.txt

********************************************************************************

SQL ID:42nypzay2vmkz

Plan Hash:4191104944

select/*+ trace_by_leo1_session */ count(*) from leo1.leo1

call     count      cpu    elapsed       disk     query    current        rows

-------------  -------- ---------- -------------------- ----------  ----------

Parse        2     0.00       0.02          0          0          0           0

Execute      2     0.00       0.00          0          0          0           0

Fetch        4     0.04       0.04          0       2062          0           2

-------------  -------- ---------- -------------------- ----------  ----------

total        8     0.04       0.07          0      2062          0           2

Misses in librarycache during parse: 2   硬解析了2次,说明把上面的2条相同SQL汇总成了一份SQL报告,每条SQL都硬解析了1次

Optimizer mode:ALL_ROWS

Parsing user id:86

Rows     Row Source Operation   

下面是SQL实际的执行计划,也有可能与autotrace不同,可以很方便的进行比对研究

------- ---------------------------------------------------

      1 SORT AGGREGATE (cr=1031 pr=0 pw=0 time=0 us)

  72007  TABLE ACCESS FULL LEO1 (cr=1031 pr=0 pw=0 time=477183 us cost=288 size=0card=72007)

********************************************************************************

SQL ID:afbr2j5352vxy

Plan Hash:908401892

select/*+ trace_by_leo1_session */ object_type,count(*) from leo1.leo1 group byobject_type

call     count      cpu    elapsed       disk     query    current        rows

-------------  -------- ---------- -------------------- ----------  ----------

Parse        1     0.00       0.00          0          0          0           0

Execute      1     0.00       0.00          0          0          0           0

Fetch        4     0.03       0.03          0       1031          0          43

-------------  -------- ---------- -------------------- ----------  ----------

total        6     0.03       0.04          0      1031          0          43

Misses in librarycache during parse: 1       这条语句也做了1次硬解析

Optimizer mode:ALL_ROWS

Parsing user id:86

Rows     Row Source Operation           这个行源操作,就是SQL实际的执行过程

------- ---------------------------------------------------

     43 HASH GROUP BY (cr=1031 pr=0 pw=0 time=0 us cost=290 size=387 card=43)

  72007  TABLE ACCESS FULL LEO1 (cr=1031 pr=0 pw=0 time=497135 us cost=288size=648063 card=72007)

********************************************************************************

使用10046事件对其它会话进行跟踪

146会话我们使用leo1用户操作

12会话我们使用leo2用户操作

146会话

LEO1@LEO1> selectdistinct sid from v$mystat;

       SID

-----------------

       146

查询一下2个会话的ID和SERIAL

LEO1@LEO1>select sid,serial# from v$session where sid in (146,12);  

       SID   SERIAL#

---------------------------- ----

        12      4777

       146      1035

LEO1@LEO1> droptable leo2;                          清理环境

Table dropped.

LEO1@LEO1>create table leo2 as select * from dba_objects; 创建leo2来做测试区别上一个实验

Table created.

LEO1@LEO1>execute sys.dbms_system.set_ev(12,4777,10046,12,'LEO2');

12:要跟踪的会话ID

4777:会话串号

10046:事件号

12:level级别

LEO2:跟踪的用户名,当前用户可设置为空

声明:这个存储过程是SYS用户特有的,所以在引用时必须带schema,不带就会报错如下

ERROR atline 1:

ORA-06550:line 1, column 7:

PLS-00201:identifier 'DBMS_SYSTEM.SET_EV' must be declared

ORA-06550:line 1, column 7:

PL/SQL:Statement ignored

12会话

LEO2@LEO1>select /*+ 10046_trace_leo2_session */ count(*) from leo1.leo2;

  COUNT(*)

----------

     72007

/*+10046_trace_leo2_session */用来标识使用10046事件追踪leo用户的会话,方便在trace文件里找

LEO2@LEO1>select /*+ 10046_trace_leo2_session */ object_type,count(*) from leo1.leo2group by object_type;

OBJECT_TYPE           COUNT(*)

-------------------------------------- ----------

EDITION                   1

INDEXPARTITION            131

CONSUMERGROUP          25

SEQUENCE                 223

TABLEPARTITION            128

SCHEDULE                  3

QUEUE                     35

RULE                       1

JAVA DATA                  328

PROCEDURE                 160

OPERATOR                  55

146会话

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_16565.trc

查询一下当前会话写入的trace文件,是不是还是找不到呢,嗯你猜对了

[oracle@leonarding1trace]$ ll | grep LEO1_ora_16565.trc   就像于谦唱的一无所有

LEO1@LEO1> exit                         leo1用户退出

Disconnected fromOracle Database 11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production

With thePartitioning, OLAP, Data Mining and Real Application Testing options

LEO2@LEO1> exit                         leo2用户退出

Disconnected fromOracle Database 11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production

With thePartitioning, OLAP, Data Mining and Real Application Testing options

[oracle@leonarding1trace]$ ll | grep LEO1_ora_16565.trc    肿么还是木有呢,看来还滴猜

我这块的trace文件没有生成,我查询了临近的所有原始文件都没有找到,知道原委的朋友可请教哦有加分滴!

 

三 演示10046 level 1,4,8,12的区别

10046事件不同级别:追踪的信息不同

Level 1:等同于SQL_TRACE

Level 4:在Level1的基础上增加收集绑定变量的信息

Level 8:在Level1的基础上增加等待事件的信息,这个有用,如果一条SQL语句非常非常慢,可以查一下是什么原因导致的如此慢

Level 12:等同于Level 4+Level 8,即同时收集绑定变量信息和等待事件信息

汇总:级别高的概括级别低的

实验

10046 Level 1

这个级别的trace信息和SQL_TRACE中的内容是一致的,我们可以用一个绑定变量SQL来测试一下效果

LEO1@LEO1> setlinesize 300                     调整格式

LEO1@LEO1> setpagesize 999

LEO1@LEO1>variable i number;                   定义变量

LEO1@LEO1>execute :i:=100;                     变量赋值

PL/SQL proceduresuccessfully completed.

LEO1@LEO1>alter session set events '10046 trace name context forever,level 1';   

Oracle 下所有启动事件的通用格式,级别为1

Session altered.

LEO1@LEO1>select * from leo2 where object_id=:i;   绑定变量,重用SQL执行计划

OWNER                          OBJECT_NAME                                                                                                                     SUBOBJECT_NAME                  OBJECT_ID DATA_OBJECT_IDOBJECT_TYPE         CREATED   LAST_DDL_ TIMESTAMP           STATUS  T G S

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- ---------- -------------- ---------------------------- --------- ------------------- ------- - - -

NAMESPACE EDITION_NAME

---------- ------------------------------

SYS                            ORA$BASE                                                                                                                                     100                 EDITION             15-AUG-09 15-AUG-092009-08-15:00:16:54 VALID   N N N

        64

LEO1@LEO1>alter session set events '10046 trace name context off';    关闭10046事件

Session altered.

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc

当前会话写入的trace文件

LEO1@LEO1> !vim/u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc  我们看一下原始trace文件

====================================================================================

PARSING IN CURSOR#2 len=37 dep=0 uid=85 oct=3 lid=85 tim=1360372361916503 hv=21593726ad='7da84c28' sqlid='4z9c6ch0nkzmy'

select *from leo2 where object_id=:i     我们看不到绑定变量值,只能看到它是一个绑定变量

END OF STMT

PARSE #2:c=2999,e=7530,p=0,cr=0,cu=0,mis=1,r=0,dep=0,og=1,plh=0,tim=1360372361916485

EXEC#2:c=2999,e=2920,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2258638698,tim=1360372361920321

FETCH#2:c=0,e=458,p=0,cr=5,cu=0,mis=0,r=1,dep=0,og=1,plh=2258638698,tim=1360372361928304

FETCH#2:c=20996,e=22661,p=0,cr=1027,cu=0,mis=0,r=0,dep=0,og=1,plh=2258638698,tim=1360372361953093

STAT #2 id=1 cnt=1pid=0 pos=1 obj=74150 op='TABLE ACCESS FULL LEO2 (cr=1032 pr=0 pw=0 time=0 uscost=288 size=97 card=1)'

*** 2013-02-0909:13:00.053

CLOSE #2:c=0,e=66,dep=0,type=0,tim=1360372380053747

使用的是2号游标,PARSE #2 解析 -> EXEC #2 执行 ->  FETCH #2  取操作 -> CLOSE #2关闭游标

SQL_TRACE和10046事件内容是按时间顺序打印出来的,过程如上所示

mis=1 我们进行了一次硬解析

在Level 1 下是看不到绑定变量值是多少的,只能看到有存在绑定变量

=====================================================================================

10046 Level 4

在这个级别中我们就可以看到绑定变量信息了

LEO1@LEO1>alter session set events '10046 trace name context forever,level 4';  我们设置级别为4

Session altered.

LEO1@LEO1>select * from leo2 where object_id=:i;   继续绑定变量

OWNER                          OBJECT_NAME                                                                                                                     SUBOBJECT_NAME                 OBJECT_ID DATA_OBJECT_ID OBJECT_TYPE         CREATED   LAST_DDL_ TIMESTAMP           STATUS T G S

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- ---------- -------------- ------------------- ------------------ ------------------- ------- - - -

NAMESPACE EDITION_NAME

----------------------------------------

SYS                            ORA$BASE                                                                                                                                     100                 EDITION             15-AUG-09 15-AUG-092009-08-15:00:16:54 VALID   N N N

        64

LEO1@LEO1>alter session set events '10046 trace name context off';    关闭10046事件

Session altered.

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc

还是同一个trace文件

LEO1@LEO1> !vim/u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc  打开trace瞧一瞧

=======================================================================================

PARSING IN CURSOR#4 len=37 dep=0 uid=85 oct=3 lid=85 tim=1360373199428632 hv=21593726ad='7da84c28' sqlid='4z9c6ch0nkzmy'

select *from leo2 where object_id=:i

END OF STMT

PARSE#4:c=0,e=254,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2258638698,tim=1360373199428619

BINDS#4:                绑定的是4号游标

Bind#0     0表示绑定的第一个变量  如果是1表示绑定的第二个变量,以此类推

  oacdty=02 mxl=22(22) mxlc=00 mal=00 scl=00pre=00

  oacflg=03 fl2=1000000 frm=00 csi=00 siz=24off=0

  kxsbbbfp=7f2e8344efa0  bln=22 avl=02  flg=05

  value=100 这就是绑定的变量值

EXEC#4:c=2000,e=31957,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2258638698,tim=1360373199461116

FETCH#4:c=1000,e=730,p=0,cr=5,cu=0,mis=0,r=1,dep=0,og=1,plh=2258638698,tim=1360373199462378

FETCH#4:c=25996,e=26510,p=0,cr=1027,cu=0,mis=0,r=0,dep=0,og=1,plh=2258638698,tim=1360373199491356

STAT #4 id=1 cnt=1pid=0 pos=1 obj=74150 op='TABLE ACCESS FULL LEO2 (cr=1032 pr=0 pw=0 time=0 uscost=288 size=97 card=1)'

*** 2013-02-0909:26:48.104

CLOSE#4:c=0,e=66,dep=0,type=1,tim=1360373208104843

mis=0   由于刚才我们已经硬解析过了,这次直接软解析,即SQL执行计划重用

这次使用的是4号游标,这个游标号是可以重复使用的,级别4就增加了绑定变量信息

=======================================================================================

10046 Level 8

这个级别中,可以显示SQL语句等待事件,我们就可以知道SQL语句是在等待某些资源,还是没有等待资源速度就是这样了,只能提升硬件

LEO1@LEO1>alter session set events '10046 trace name context forever,level 8';  级别为8

Session altered.

LEO1@LEO1>select count(*) from leo2 where object_id=:i group by object_name; 执行SQL

  COUNT(*)

----------

         1

LEO1@LEO1>alter session set events '10046 trace name context off';           关闭10046事件

Session altered.

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc

LEO1@LEO1> !vim/u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc   我们定眼观瞧

======================================================================================

PARSING IN CURSOR#4 len=65 dep=0 uid=85 oct=3 lid=85 tim=1360373900654762 hv=3813172666ad='7dbb2f78' sqlid='fr4tufgjnhtdu'

selectcount(*) from leo2 where object_id=:i group by object_name

END OF STMT

PARSE#4:c=1999,e=1923,p=0,cr=0,cu=0,mis=1,r=0,dep=0,og=1,plh=0,tim=1360373900654751

EXEC#4:c=5000,e=5370,p=0,cr=0,cu=0,mis=1,r=0,dep=0,og=1,plh=2474819896,tim=1360373900660419

WAIT #4:nam='SQL*Net message to client'ela= 17 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360373900661831

FETCH #4:c=20997,e=20590,p=0,cr=1031,cu=0,mis=0,r=1,dep=0,og=1,plh=2474819896,tim=1360373900682600

WAIT #4:nam='SQL*Net message from client'ela= 1168 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360373900684127

FETCH#4:c=1000,e=245,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2474819896,tim=1360373900684583

STAT #4 id=1 cnt=1pid=0 pos=1 obj=0 op='HASH GROUP BY (cr=1031 pr=0 pw=0 time=0 us cost=289size=30 card=1)'

STAT #4 id=2 cnt=1pid=1 pos=1 obj=74150 op='TABLE ACCESS FULL LEO2 (cr=1031 pr=0 pw=0 time=0 uscost=288 size=30 card=1)'

WAIT #4:nam='SQL*Net message to client' ela= 14 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360373900685343

*** 2013-02-0909:38:28.482

WAIT #4:nam='SQL*Net message from client' ela= 7797435 driver id=1650815232 #bytes=1p3=0 obj#=-1 tim=1360373908482852

CLOSE#4:c=0,e=90,dep=0,type=0,tim=1360373908483370

这次还是重用4号游标,标红的都是标识SQL语句WAIT等待事件,这几个等待事件都是正常的等待客户端发送命令的事件,说明我们的SQL语句效率都挺高的

======================================================================================

10046 Level 12

这个级别即可以显示绑定变量,又可以显示等待事件

LEO1@LEO1>alter session set events '10046 trace name context forever,level 12';   现在级别可为12

Session altered.

LEO1@LEO1>select count(*) from leo2 where object_id=:i group by object_name;

  COUNT(*)

----------

         1

LEO1@LEO1>alter session set events '10046 trace name context off';     关闭10046事件

Session altered.

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc

LEO1@LEO1> !vim/u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc   我们再次观瞧

==================================================================================

PARSING IN CURSOR#2 len=65 dep=0 uid=85 oct=3 lid=85 tim=1360375064416180 hv=3813172666ad='7dbb2f78' sqlid='fr4tufgjnhtdu'

select count(*)from leo2 where object_id=:i group by object_name

END OF STMT

PARSE#2:c=0,e=266,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2474819896,tim=1360375064416167

BINDS#2:                这是绑定变量的信息

Bind#0                0表示绑定的第一个变量

  oacdty=02 mxl=22(22) mxlc=00 mal=00 scl=00pre=00

  oacflg=03 fl2=1000000 frm=00 csi=00 siz=24off=0

  kxsbbbfp=7f2e834ccec8  bln=22 avl=02  flg=05

  value=100             变量值

EXEC#2:c=1000,e=639,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2474819896,tim=1360375064417345

WAIT #2:nam='SQL*Net message to client' ela= 18 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360375064417579

FETCH#2:c=18997,e=20107,p=0,cr=1031,cu=0,mis=0,r=1,dep=0,og=1,plh=2474819896,tim=1360375064437826

WAIT #2:nam='SQL*Net message from client' ela= 846 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360375064439398

FETCH#2:c=0,e=127,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2474819896,tim=1360375064439680

STAT #2 id=1 cnt=1pid=0 pos=1 obj=0 op='HASH GROUP BY (cr=1031 pr=0 pw=0 time=0 us cost=289size=30 card=1)'

STAT #2 id=2 cnt=1pid=1 pos=1 obj=74150 op='TABLE ACCESS FULL LEO2 (cr=1031 pr=0 pw=0 time=0 uscost=288 size=30 card=1)'

WAIT #2:nam='SQL*Net message to client' ela= 18 driver id=1650815232 #bytes=1 p3=0 obj#=-1 tim=1360375064440089

又有等待事件的信息,也是我们常用的级别

*** 2013-02-0909:57:52.607

WAIT #2:nam='SQL*Net message from client' ela= 8167539 driver id=1650815232 #bytes=1p3=0 obj#=-1 tim=1360375072607728

CLOSE#2:c=0,e=129,dep=0,type=0,tim=1360375072608199

这次使用的是2号游标,PARSE #2 马上 EXEC #2 执行 FETCH #2 取操作 紧接着输出执行计划

==================================================================================

小结:到此我们完成了1~12级别的对比说明,10046事件是我们非常常用的优化SQL的监控工具,而我们最常用的是级别12,有了这些信息我们就可以游刃有余的分析和改善SQL性能

 

 

四 自己构造两条完全同样功能的SQL,通过对SQL_TRACE产生的trace文件的分析,比较它们的性能忧劣。

LEO1@LEO1> droptable leo4 purge;                          清空环境

Table dropped.

LEO1@LEO1>create table leo4 as select * from dba_objects;      创建leo4表

Table created.

LEO1@LEO1> altersession set sql_trace=true;     启动SQL_TRACE

Session altered.

LEO1@LEO1>select count(*) from leo4;          功能:统计全表总记录数

  COUNT(*)

---------------------

     71968

LEO1@LEO1>select sum(count_id) from (select count(object_id) count_id from leo4 group byobject_type);

SUM(COUNT_ID)          功能一样但消耗的资源不同,说明了SQL功能性和优化性同等重要

----------------------

        71968

LEO1@LEO1>alter session set sql_trace=false;     关闭SQL_TRACE

Session altered.

LEO1@LEO1>select name,value from v$diag_info where name='Default Trace File';

NAME             VALUE

----------------------------------------------------------------------------------------------------------------------------------------------

Default Trace File    /u01/app/oracle/diag/rdbms/leo1/LEO1/trace/LEO1_ora_25998.trc

[oracle@leonarding1trace]$ tkprof LEO1_ora_25998.trc 25998_2.txt

TKPROF: Release11.2.0.1.0 - Development on Tue Feb 12 17:38:34 2013

Copyright (c)1982, 2009, Oracle and/or its affiliates. All rights reserved.

 

********************************************************************************

SQL ID:7hruqqqzx2zs7

Plan Hash:3210696650

selectcount(*) from leo4

call     count      cpu    elapsed       disk     query    current        rows

-------------  -------- ---------- -------------------- ----------  ----------

Parse        1     0.00       0.00          0          1          0           0

Execute      1     0.00       0.00          0          0          0           0

Fetch        2     0.03       0.03        722       1030          0           1

-------------  -------- ---------- -------------------- ----------  ----------

total        4     0.03       0.03        722      1031          0           1

Misses in librarycache during parse: 1        

Optimizer mode:ALL_ROWS

Parsing user id:85

Rows     Row Source Operation

------- ---------------------------------------------------

      1 SORT AGGREGATE (cr=1030 pr=722 pw=0 time=0 us)

  71968  TABLE ACCESS FULL LEO4 (cr=1030 pr=722 pw=0 time=825511 us cost=287size=0 card=77424)

 

1.有1次硬解析

2.cpu time 和 elapsed time   0.03

3.行源操作只有2步就可完成

4.cr=1030  一个指标

5.由于是第一次统计,所以产生了pr=722次物理读

***************************************************************************************

SQL ID:9zhvsut63s579

Plan Hash:2452925700

selectsum(count_id) from (select count(object_id) count_id from leo4 group byobject_type)

call     count      cpu    elapsed       disk     query    current        rows

-------------  -------- ---------- -------------------- ----------  ----------

Parse        1     0.00       0.00          0          1          0           0

Execute      1     0.00       0.00          0          0          0           0

Fetch        2     0.04       0.04          0       1030          0           1

-------------  -------- ---------- -------------------- ----------  ----------

total        4     0.04       0.04          0      1031          0           1

Misses in librarycache during parse: 1

Optimizer mode:ALL_ROWS

Parsing user id:85

Rows     Row Source Operation

------- ---------------------------------------------------

      1 SORT AGGREGATE (cr=1030 pr=0 pw=0 time=0 us)

     43  VIEW  (cr=1030 pr=0 pw=0 time=588us cost=290 size=1006512 card=77424)

     43   HASH GROUP BY (cr=1030 pr=0 pw=0 time=210 us cost=290 size=1858176card=77424)

  71968    TABLE ACCESS FULL LEO4 (cr=1030 pr=0 pw=0 time=694104 us cost=287size=1858176 card=77424)

 

1.有1次硬解析

2.cpu time 和 elapsed time   0.04

3.行源操作需要处理4步才能完成,因此所用时间就会较长,效率较低

4.cr=1030  四个指标

5.由于是第二次统计,所以没有物理读,但所用的耗时却多

***************************************************************************************

 

©著作权归作者所有:来自51CTO博客作者leonarding1的原创作品,如需转载,请注明出处,否则将追究法律责任

OracleSQLSQL_TRACE性能优化


打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP