oracle固定执行计划

阅读 75

2023-06-18

文档课题:oracle固定执行计划.
本文介绍oracle固定执行计划的2种方法,outline,SQL Profile,其中SQL Profile是重点。
1、outline
1.1、相关概念
开发环境中SQL的执行无问题,但在生产环境执行效率却异常缓慢。若此时更改SQL需重新修改及编译源程序,成本非常高。此时可使用outline在不改变原应用程序的情况下更改特定SQL的执行计划。
outline的原理是将调好的SQL执行计划(一系列HINT)存储起来,然后将该执行计划所对应的SQL用目前系统效率低下的SQL替代,从而达到系统每次执行该SQL时都使用已存储的执行计划。
当发现低效SQL,可使用hint对其优化;对于SQL代码可以修改的场景,直接修改SQL代码加上hint即可;对于SQL代码不可修改的情况,Oracle提供outline功能为SQL修改hint。
使用 outLine步骤:
A、生成新SQL和老SQL的2个outline
B、交换两个SQL的提示信息
C、ON LOGON触发器设定session的CATEGORY(自定义类别)
1.2、outline测试过程
1.2.1、数据准备
LEO@orcl150> select * from v$version;

BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
PL/SQL Release 11.2.0.4.0 - Production
CORE    11.2.0.4.0      Production
TNS for Linux: Version 11.2.0.4.0 - Production
NLSRTL Version 11.2.0.4.0 – Production

LEO@orcl150> col username for a15
LEO@orcl150> col granted_role for a15
LEO@orcl150> select * from user_role_privs;

USERNAME        GRANTED_ROLE    ADM DEF OS_
--------------- --------------- --- --- ---
LEO             CONNECT         NO  YES NO
LEO             DBA             NO  YES NO
LEO@orcl150> select * from session_privs where privilege like '%OUTLINE%';

PRIVILEGE
----------------------------------------
CREATE ANY OUTLINE
ALTER ANY OUTLINE
DROP ANY OUTLINE

SYS@orcl150> grant all on ol$hints to leo;

Grant succeeded.

SYS@orcl150> conn leo/leo;
Connected.

LEO@orcl150> create table tb_leo_20230305 as select * from dba_tables;

Table created.

LEO@orcl150> create index idx_tb_leo_20230305 on tb_leo_20230305(table_name);

Index created.

LEO@orcl150> set autotrace on
LEO@orcl150> select owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

no rows selected


Execution Plan
----------------------------------------------------------
Plan hash value: 3703795454

---------------------------------------------------------------------------------------------------
| Id  | Operation                   | Name                | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT            |                     |     1 |    34 |     1   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID| TB_LEO_20230305     |     1 |    34 |     1   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN          | IDX_TB_LEO_20230305 |     1 |       |     1   (0)| 00:00:01 |
---------------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("TABLE_NAME"='TB_LEO_20230305')

Note
-----
   - dynamic sampling used for this statement (level=2)


Statistics
----------------------------------------------------------
         11  recursive calls
          0  db block gets
         71  consistent gets
          1  physical reads
          0  redo size
        333  bytes sent via SQL*Net to client
        513  bytes received via SQL*Net from client
          1  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
0	rows processed
LEO@orcl150> select /*+full(TB_LEO_20230305)*/ owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

no rows selected


Execution Plan
----------------------------------------------------------
Plan hash value: 2133804736

-------------------------------------------------------------------------------------
| Id  | Operation         | Name            | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                 |     1 |    34 |    31   (0)| 00:00:01 |
|*  1 |  TABLE ACCESS FULL| TB_LEO_20230305 |     1 |    34 |    31   (0)| 00:00:01 |
-------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter("TABLE_NAME"='TB_LEO_20230305')

Note
-----
   - dynamic sampling used for this statement (level=2)


Statistics
----------------------------------------------------------
          7  recursive calls
          0  db block gets
        168  consistent gets
          0  physical reads
          0  redo size
        333  bytes sent via SQL*Net to client
        513  bytes received via SQL*Net from client
          1  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
0	rows processed
LEO@orcl150> set autotrace off;
1.2.2、创建outline
--分别给索引扫描与全表扫描创建各自的outline
LEO@orcl150> create or replace outline tb_leo_20230305_l on select owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

Outline created.

LEO@orcl150> create or replace outline tb_leo_20230305_m on select /*+full(TB_LEO_20230305)*/ owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

Outline created.
1.2.3、确认outline
LEO@orcl150> col sql_text for a80
LEO@orcl150> set long 2000
LEO@orcl150> select name,used,sql_text from dba_outlines where name like '%TB_LEO_20230305%';

NAME                 USED   SQL_TEXT
-------------------- ------ ----------------------------------------------------------------------------------------------------
TB_LEO_20230305_L    UNUSED select owner from tb_leo_20230305 where table_name='TB_LEO_20230305'
TB_LEO_20230305_M    UNUSED select /*+full(TB_LEO_20230305)*/ owner from tb_leo_20230305 where table_name='TB_LEO_20230305'
1.2.4、互换执行计划
--更新outln.ol$表,达到执行计划的互换.
LEO@orcl150> update outln.ol$ set ol_name=decode(ol_name,'TB_LEO_20230305_M','TB_LEO_20230305_L','TB_LEO_20230305_L','TB_LEO_20230305_M') where ol_name in ('TB_LEO_20230305_L','TB_LEO_20230305_M');

2 rows updated.

LEO@orcl150> commit;

Commit complete.

LEO@orcl150> select name,used,sql_text from dba_outlines where name like '%TB_LEO_20230305%';

NAME                 USED   SQL_TEXT
-------------------- ------ ----------------------------------------------------------------------------------------------------
TB_LEO_20230305_L    UNUSED select /*+full(TB_LEO_20230305)*/ owner from tb_leo_20230305 where table_name='TB_LEO_20230305'
TB_LEO_20230305_M    UNUSED select owner from tb_leo_20230305 where table_name='TB_LEO_20230305'
LEO@orcl150> col hint for a90
LEO@orcl150> select name,hint from dba_outline_hints where join_pos=1 and name like '%TB_LEO_20230305%';

NAME                 HINT
-------------------- ------------------------------------------------------------------------------------------
TB_LEO_20230305_L    INDEX_RS_ASC(@"SEL$1" "TB_LEO_20230305"@"SEL$1" ("TB_LEO_20230305"."TABLE_NAME"))
TB_LEO_20230305_M    FULL(@"SEL$1" "TB_LEO_20230305"@"SEL$1")
1.2.5、互换后验证
--执行计划互换后,对sql语句进行验证
LEO@orcl150> set autotrace on
LEO@orcl150> alter system set use_stored_outlines=true;

System altered.

LEO@orcl150> select owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

no rows selected


Execution Plan
----------------------------------------------------------
Plan hash value: 2133804736

-------------------------------------------------------------------------------------
| Id  | Operation         | Name            | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                 |    88 |  2992 |    31   (0)| 00:00:01 |
|*  1 |  TABLE ACCESS FULL| TB_LEO_20230305 |    88 |  2992 |    31   (0)| 00:00:01 |
-------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter("TABLE_NAME"='TB_LEO_20230305')

Note
-----
   - outline "TB_LEO_20230305_M" used for this statement


Statistics
----------------------------------------------------------
         36  recursive calls
        148  db block gets
        124  consistent gets
          1  physical reads
        624  redo size
        333  bytes sent via SQL*Net to client
        513  bytes received via SQL*Net from client
          1  SQL*Net roundtrips to/from client
          2  sorts (memory)
          0  sorts (disk)
0	rows processed

说明:如上所示,执行计划互换后,原本应走索引范围扫描变成全表扫描。
LEO@orcl150> select /*+full(TB_LEO_20230305)*/ owner from tb_leo_20230305 where table_name='TB_LEO_20230305';

no rows selected


Execution Plan
----------------------------------------------------------
Plan hash value: 3703795454

---------------------------------------------------------------------------------------------------
| Id  | Operation                   | Name                | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT            |                     |    88 |  2992 |     7   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID| TB_LEO_20230305     |    88 |  2992 |     7   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN          | IDX_TB_LEO_20230305 |    35 |       |     1   (0)| 00:00:01 |
---------------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("TABLE_NAME"='TB_LEO_20230305')

Note
-----
   - outline "TB_LEO_20230305_L" used for this statement


Statistics
----------------------------------------------------------
         34  recursive calls
        147  db block gets
         23  consistent gets
          0  physical reads
        584  redo size
        333  bytes sent via SQL*Net to client
        513  bytes received via SQL*Net from client
          1  SQL*Net roundtrips to/from client
          2  sorts (memory)
          0  sorts (disk)
          0  rows processed
说明:如上所示,执行计划互换后,原本应该是全表扫描的变成索引范围扫描.
2、sql profile
2.1、相关概念
oracle 11g后续版本参数use_stored_outlines已不存在。意味着不能使用create outline方式来为一个sql创建hint,以便使用store outline固定执行计划。SQL Profile就是为某一SQL语句提供除系统统计信息、对象(表和索引等)统计信息之外的其他信息(比如运行环境、额外的更准确的统计信息),以帮助优化器为SQL语句选择更适合的执行计划。Outlines能实现的功能SQL Profiles完全能够实现,而且SQL Profiles具有Outlines不具备的优点,最重要的有以下两点: 
A、SQL Profiles更容易生成、更改和控制;
B、SQL Profiles在对SQL语句的支持上做得更好,适用范围更广。 
10g之前有outlines,10g之后sql profile作为新特性之一出现。如果针对非绑定变量的sql,outlines会力不从心。sql profile最大的优点是在不修改sql语句和会话运行环境的情况下去优化sql的执行效率,适合无法在应用程序中修改sql语句的场景.
SQL Profile对以下类型语句有效:
SELECT语句;
UPDATE语句;
INSERT语句(仅当使用SELECT子句时有效);
DELETE语句;
CREATE语句(仅当使用SELECT子句时有效);
MERGE语句(仅当作UPDATE和INSERT操作时有效)。
另外,使用SQL Profile还必须有CREATE ANY SQL PROFILE、DROP ANY SQL PROFILE和ALTER ANY SQL PROFILE等系统权限。
2.2、SQL Profile测试
说明:有两种生成SQL Profile的方法,分别为手动和STA。
2.2.1、手工创建SQL Profile 方法1
2.2.1.1、准备数据
--手工创建SQL Profile
创建测试表,根据DBA_OBJECTS创建,OBJECT_ID上有索引。
LEO@orcl150> select * from v$version;

BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
PL/SQL Release 11.2.0.4.0 - Production
CORE    11.2.0.4.0      Production
TNS for Linux: Version 11.2.0.4.0 - Production
NLSRTL Version 11.2.0.4.0 – Production
LEO@orcl150> drop table tb_leo_20230305 purge;

Table dropped.

LEO@orcl150> create table tb_leo_20230305 as select * from dba_objects;

Table created.

LEO@orcl150> create index ind_tb_leo_id on tb_leo_20230305(object_id);

Index created.
--查看sql的默认执行计划,走的索引,通过指定outline可以获取系统为sql生成的hint
LEO@orcl150> explain plan for select * from tb_leo_20230305 where object_id=:a;

Explained.
LEO@orcl150> select * from table(dbms_xplan.display(null,null,'outline'));

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------
Plan hash value: 3587901987

-----------------------------------------------------------------------------------------------
| Id  | Operation                   | Name            | Rows  | Bytes | Cost (%CPU)| Time     |
-----------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT            |                 |   815 |   164K|     7   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID| TB_LEO_20230305 |   815 |   164K|     7   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN          | IND_TB_LEO_ID   |   326 |       |     1   (0)| 00:00:01 |
-----------------------------------------------------------------------------------------------

Outline Data

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------
  /*+
      BEGIN_OUTLINE_DATA
      INDEX_RS_ASC(@"SEL$1" "TB_LEO_20230305"@"SEL$1" ("TB_LEO_20230305"."OBJECT_ID"))
      OUTLINE_LEAF(@"SEL$1")
      ALL_ROWS
      DB_VERSION('11.2.0.4')
      OPTIMIZER_FEATURES_ENABLE('11.2.0.4')
      IGNORE_OPTIM_EMBEDDED_HINTS
      END_OUTLINE_DATA

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------
  */

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("OBJECT_ID"=TO_NUMBER(:A))

Note
   - dynamic sampling used for this statement (level=2)

32 rows selected.
2.2.1.2、生成全表扫描hint
--如果想让其走全表扫描,首先获取全表扫描hint。
LEO@orcl150> explain plan for select /*+ full(TB_LEO_20230305) */ * from tb_leo_20230305 where object_id=:a;

Explained.
LEO@orcl150> select * from table(dbms_xplan.display(null,null,'outline'));

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------
Plan hash value: 2133804736

-------------------------------------------------------------------------------------
| Id  | Operation         | Name            | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                 |   815 |   164K|   345   (1)| 00:00:05 |
|*  1 |  TABLE ACCESS FULL| TB_LEO_20230305 |   815 |   164K|   345   (1)| 00:00:05 |
-------------------------------------------------------------------------------------

Outline Data
-------------

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------

  /*+
      BEGIN_OUTLINE_DATA
      FULL(@"SEL$1" "TB_LEO_20230305"@"SEL$1")
      OUTLINE_LEAF(@"SEL$1")
      ALL_ROWS
      DB_VERSION('11.2.0.4')
      OPTIMIZER_FEATURES_ENABLE('11.2.0.4')
      IGNORE_OPTIM_EMBEDDED_HINTS
      END_OUTLINE_DATA
  */

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter("OBJECT_ID"=TO_NUMBER(:A))

Note
-----
   - dynamic sampling used for this statement (level=2)

31 rows selected.
说明:可以看到全面扫描的hint已经生成。
2.2.1.3、创建sql profile
--根据生成的hint信息,选取必要的数据,创建sql profile。
LEO@orcl150> declare v_hints sys.sqlprof_attr;
  2  begin
  3  v_hints := sys.sqlprof_attr('FULL(@"SEL$1" "TB_LEO_20230305"@"SEL$1")');   --从上面outline data部分获取到hint
  4  dbms_sqltune.import_sql_profile('select * from TB_LEO_20230305 where object_id= :a',  --sql语句部分
  5  v_hints,
  6  'TB_LEO_20230305',   --profile的名字
  7  force_match => true);
  8  end;
  9  /

PL/SQL procedure successfully completed.
2.2.1.4、验证执行计划
--查看是否生效。
LEO@orcl150> explain plan for select * from tb_leo_20230305 where object_id=:a;

Explained.

LEO@orcl150> select * from table(dbms_xplan.display);

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------
Plan hash value: 2133804736

-------------------------------------------------------------------------------------
| Id  | Operation         | Name            | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                 |   815 |   164K|   345   (1)| 00:00:05 |
|*  1 |  TABLE ACCESS FULL| TB_LEO_20230305 |   815 |   164K|   345   (1)| 00:00:05 |
-------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

PLAN_TABLE_OUTPUT
----------------------------------------------------------------------------------------------------

   1 - filter("OBJECT_ID"=TO_NUMBER(:A))

Note
-----
   - dynamic sampling used for this statement (level=2)
   - SQL profile "TB_LEO_20230305" used for this statement

18 rows selected.
LEO@orcl150> SELECT b.name, d.sql_text, extractvalue(value(h), '.') as hints
  2    FROM dba_sql_profiles d,
  3         SYS.SQLOBJ$DATA  a,
  4         SYS.SQLOBJ$ b,
  5         TABLE(XMLSEQUENCE(EXTRACT(XMLTYPE(A.COMP_DATA),'/outline_data/hint'))) h
  6  where a.signature=b.signature
  7    and a.category=b.category
  8    and a.obj_type=b.obj_type
  9    and a.plan_id=b.plan_id
 10    and a.signature=d.signature
 11    and d.name='TB_LEO_20230305';

NAME                 SQL_TEXT                                           HINTS
-------------------- -------------------------------------------------- --------------------------------------------------
TB_LEO_20230305      select * from TB_LEO_20230305 where object_id= :a  FULL(@"SEL$1" "TB_LEO_20230305"@"SEL$1")
说明:如上所示,原本应该走索引范围扫描的语句,使用sql profile修改执行计划后变为全表扫描.
2.2.2、手工创建SQL Profile 方法2
使用coe_xfr_sql_profile.sql脚本生成sqlprof_attr数据,比较麻烦的是sqlprof_attr('FULL(t1@SEL$1)')格式的编辑。Mos Note 215187.1中sqlt.zip的目录utl提供脚本coe_xfr_sql_profile.sql可以生成这些信息。
说明:建议使用该方法固定执行计划.
2.2.2.1、数据准备
LEO@orcl150> create table scott.test as select * from dba_objects;

Table created.

LEO@orcl150> create index scott.idx_test_01 on scott.test(object_id);

Index created.

LEO@orcl150> exec dbms_stats.gather_table_stats('scott','test',cascade=>true);

PL/SQL procedure successfully completed.

LEO@orcl150> update scott.test set object_id=10 where object_id>10;

86419 rows updated.

LEO@orcl150> commit;

Commit complete.
LEO@orcl150> select object_id,count(*) from scott.test group by object_id;

 OBJECT_ID   COUNT(*)
---------- ----------
         6          1
         2          1
         5          1
         4          1
         8          1
         3          1
         7          1
        10      86420
         9          1

9 rows selected.
2.2.2.2、执行查询
--查询1
--执行查询,发现执行计划走的是索引范围扫描。
LEO@orcl150> set autot traceonly explain stat
LEO@orcl150> select * from scott.test where object_id=10;

86420 rows selected.


Execution Plan
----------------------------------------------------------
Plan hash value: 2317948335

-------------------------------------------------------------------------------------------
| Id  | Operation                   | Name        | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT            |             |     1 |    98 |     2   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID| TEST        |     1 |    98 |     2   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN          | IDX_TEST_01 |     1 |       |     1   (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("OBJECT_ID"=10)


Statistics
----------------------------------------------------------
          1  recursive calls
          0  db block gets
      14454  consistent gets
        905  physical reads
     110392  redo size
    9773775  bytes sent via SQL*Net to client
      63895  bytes received via SQL*Net from client
       5763  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
      86420  rows processed

说明:实际上此时表中大部分行的object_id已被更新为10,走索引已然是不合理的.
--查询2,加全表扫描hint进行查询
LEO@orcl150> select /*+ full(test)*/ * from scott.test where object_id=10;

86420 rows selected.


Execution Plan
----------------------------------------------------------
Plan hash value: 1357081020

--------------------------------------------------------------------------
| Id  | Operation         | Name | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |      |     1 |    98 |   345   (1)| 00:00:05 |
|*  1 |  TABLE ACCESS FULL| TEST |     1 |    98 |   345   (1)| 00:00:05 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter("OBJECT_ID"=10)


Statistics
----------------------------------------------------------
          1  recursive calls
          0  db block gets
       6908  consistent gets
          0  physical reads
          0  redo size
    4056770  bytes sent via SQL*Net to client
      63895  bytes received via SQL*Net from client
       5763  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
      86420  rows processed
2.2.2.3、查sql_id,plan_hash_value
--查询如上两个select语句的sql_id,plan_hash_value。
LEO@orcl150> set autot off
LEO@orcl150> col sql_text format a100
LEO@orcl150> col sql_text for a50
LEO@orcl150> select sql_text,sql_id,plan_hash_value from v$sql where sql_text like 'select * from scott.test where object_id=10%'

SQL_TEXT                                           SQL_ID        PLAN_HASH_VALUE
-------------------------------------------------- ------------- ---------------
select * from scott.test where object_id=10        cpk9jsg2qt52r      2317948335
LEO@orcl150> col sql_text for a70
LEO@orcl150> select sql_text,sql_id,plan_hash_value from v$sql where sql_text like 'select /*+ full(test)*/ * from scott.test where object_id=10%'

SQL_TEXT                                                               SQL_ID        PLAN_HASH_VALUE
---------------------------------------------------------------------- ------------- ---------------
select /*+ full(test)*/ * from scott.test where object_id=10           g82ru63zvt2yb      1357081020
2.2.2.4、处理coe_xfr_sql_profile.sql
将coe_xfr_sql_profile.sql放在$ORACLE_HOME/rdbms/admin.
 
2.2.2.5、产生COE_XFR_SQL_PROFILE
说明:运用SQL_ID和PLAN_HASH_VALUE值,对如上两个SQL生成COE_XFR_SQL_PROFILE文件.
--SQL语句1:
[oracle@leo-oel150 admin]$ sqlplus / as sysdba

SQL*Plus: Release 11.2.0.4.0 Production on Sun Mar 5 22:59:56 2023

Copyright (c) 1982, 2013, Oracle.  All rights reserved.


Connected to:
Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
With the Partitioning, OLAP, Data Mining and Real Application Testing options

SYS@orcl150> @coe_xfr_sql_profile.sql cpk9jsg2qt52r 2317948335

Parameter 1:
SQL_ID (required)



PLAN_HASH_VALUE AVG_ET_SECS
--------------- -----------
     2317948335        .089

Parameter 2:
PLAN_HASH_VALUE (required)


Values passed to coe_xfr_sql_profile:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
SQL_ID         : "cpk9jsg2qt52r"
PLAN_HASH_VALUE: "2317948335"

SQL>BEGIN
  2    IF :sql_text IS NULL THEN
  3      RAISE_APPLICATION_ERROR(-20100, 'SQL_TEXT for SQL_ID &&sql_id. was not found in memory (gv$sqltext_with_newlines) or AWR (dba_hist_sqltext).');
  4    END IF;
  5  END;
  6  /
SQL>SET TERM OFF;
SQL>BEGIN
  2    IF :other_xml IS NULL THEN
  3      RAISE_APPLICATION_ERROR(-20101, 'PLAN for SQL_ID &&sql_id. and PHV &&plan_hash_value. was not found in memory (gv$sql_plan) or AWR (dba_hist_sql_plan).');
  4    END IF;
  5  END;
  6  /
SQL>SET TERM OFF;

Execute coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql
on TARGET system in order to create a custom SQL Profile
with plan 2317948335 linked to adjusted sql_text.


COE_XFR_SQL_PROFILE completed.

--SQL语句2:
SQL>@coe_xfr_sql_profile.sql g82ru63zvt2yb 1357081020

Parameter 1:
SQL_ID (required)



PLAN_HASH_VALUE AVG_ET_SECS
--------------- -----------
     1357081020        .045

Parameter 2:
PLAN_HASH_VALUE (required)


Values passed to coe_xfr_sql_profile:
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
SQL_ID         : "g82ru63zvt2yb"
PLAN_HASH_VALUE: "1357081020"

SQL>BEGIN
  2    IF :sql_text IS NULL THEN
  3      RAISE_APPLICATION_ERROR(-20100, 'SQL_TEXT for SQL_ID &&sql_id. was not found in memory (gv$sqltext_with_newlines) or AWR (dba_hist_sqltext).');
  4    END IF;
  5  END;
  6  /
SQL>SET TERM OFF;
SQL>BEGIN
  2    IF :other_xml IS NULL THEN
  3      RAISE_APPLICATION_ERROR(-20101, 'PLAN for SQL_ID &&sql_id. and PHV &&plan_hash_value. was not found in memory (gv$sql_plan) or AWR (dba_hist_sql_plan).');
  4    END IF;
  5  END;
  6  /
SQL>SET TERM OFF;

Execute coe_xfr_sql_profile_g82ru63zvt2yb_1357081020.sql
on TARGET system in order to create a custom SQL Profile
with plan 1357081020 linked to adjusted sql_text.


COE_XFR_SQL_PROFILE completed.
2.2.2.6、修改SYS.SQLPROF_ATTR内容
说明:将coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql文件中的SYS.SQLPROF_ATTR部分更改为coe_xfr_sql_profile_g82ru63zvt2yb_1357081020.sql中的SYS.SQLPROF_ATTR部分。
2.2.2.6.1、原文件内容
---coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql中的SYS.SQLPROF_ATTR
h := SYS.SQLPROF_ATTR(
q'[BEGIN_OUTLINE_DATA]',
q'[IGNORE_OPTIM_EMBEDDED_HINTS]',
q'[OPTIMIZER_FEATURES_ENABLE('11.2.0.4')]',
q'[DB_VERSION('11.2.0.4')]',
q'[ALL_ROWS]',
q'[OUTLINE_LEAF(@"SEL$1")]',
q'[INDEX_RS_ASC(@"SEL$1" "TEST"@"SEL$1" ("TEST"."OBJECT_ID"))]',
q'[END_OUTLINE_DATA]');
---coe_xfr_sql_profile_g82ru63zvt2yb_1357081020.sql中产生的SYS.SQLPROF_ATTR
h := SYS.SQLPROF_ATTR(
q'[BEGIN_OUTLINE_DATA]',
q'[IGNORE_OPTIM_EMBEDDED_HINTS]',
q'[OPTIMIZER_FEATURES_ENABLE('11.2.0.4')]',
q'[DB_VERSION('11.2.0.4')]',
q'[ALL_ROWS]',
q'[OUTLINE_LEAF(@"SEL$1")]',
q'[FULL(@"SEL$1" "TEST"@"SEL$1")]',
q'[END_OUTLINE_DATA]');
2.2.2.6.2、系统层面修改
[oracle@leo-oel150 admin]$ vi coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql
…… (替换为全表扫描的SQLPROF_ATTR)
h := SYS.SQLPROF_ATTR(
q'[BEGIN_OUTLINE_DATA]',
q'[IGNORE_OPTIM_EMBEDDED_HINTS]',
q'[OPTIMIZER_FEATURES_ENABLE('11.2.0.4')]',
q'[DB_VERSION('11.2.0.4')]',
q'[ALL_ROWS]',
q'[OUTLINE_LEAF(@"SEL$1")]',
q'[FULL(@"SEL$1" "TEST"@"SEL$1")]',
q'[END_OUTLINE_DATA]');
……
2.2.2.7、执行修改后的COE_XFR_SQL_PROFILE
--修改后执行替换过SYS.SQLPROF_ATTR的SQL,coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql
[oracle@leo-oel150 admin]$ sqlplus / as sysdba

SQL*Plus: Release 11.2.0.4.0 Production on Sun Mar 5 23:16:07 2023

Copyright (c) 1982, 2013, Oracle.  All rights reserved.


Connected to:
Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
With the Partitioning, OLAP, Data Mining and Real Application Testing options

SYS@orcl150> @coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql
SYS@orcl150> REM
SYS@orcl150> REM $Header: 215187.1 coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql 11.4.4.4 2023/03/05 carlos.sierra $
SYS@orcl150> REM
SYS@orcl150> REM Copyright (c) 2000-2012, Oracle Corporation. All rights reserved.
SYS@orcl150> REM
SYS@orcl150> REM AUTHOR
SYS@orcl150> REM   carlos.sierra@oracle.com
SYS@orcl150> REM
SYS@orcl150> REM SCRIPT
SYS@orcl150> REM   coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql
SYS@orcl150> REM
SYS@orcl150> REM DESCRIPTION
SYS@orcl150> REM   This script is generated by coe_xfr_sql_profile.sql
SYS@orcl150> REM   It contains the SQL*Plus commands to create a custom
SYS@orcl150> REM   SQL Profile for SQL_ID cpk9jsg2qt52r based on plan hash
SYS@orcl150> REM   value 2317948335.
SYS@orcl150> REM   The custom SQL Profile to be created by this script
SYS@orcl150> REM   will affect plans for SQL commands with signature
SYS@orcl150> REM   matching the one for SQL Text below.
SYS@orcl150> REM   Review SQL Text and adjust accordingly.
SYS@orcl150> REM
SYS@orcl150> REM PARAMETERS
SYS@orcl150> REM   None.
SYS@orcl150> REM
SYS@orcl150> REM EXAMPLE
SYS@orcl150> REM   SQL> START coe_xfr_sql_profile_cpk9jsg2qt52r_2317948335.sql;
SYS@orcl150> REM
SYS@orcl150> REM NOTES
SYS@orcl150> REM   1. Should be run as SYSTEM or SYSDBA.
SYS@orcl150> REM   2. User must have CREATE ANY SQL PROFILE privilege.
SYS@orcl150> REM   3. SOURCE and TARGET systems can be the same or similar.
SYS@orcl150> REM   4. To drop this custom SQL Profile after it has been created:
SYS@orcl150> REM         EXEC DBMS_SQLTUNE.DROP_SQL_PROFILE('coe_cpk9jsg2qt52r_2317948335');
SYS@orcl150> REM   5. Be aware that using DBMS_SQLTUNE requires a license
SYS@orcl150> REM         for the Oracle Tuning Pack.
SYS@orcl150> REM   6. If you modified a SQL putting Hints in order to produce a desired
SYS@orcl150> REM         Plan, you can remove the artifical Hints from SQL Text pieces below.
SYS@orcl150> REM         By doing so you can create a custom SQL Profile for the original
SYS@orcl150> REM         SQL but with the Plan captured from the modified SQL (with Hints).
SYS@orcl150> REM
SYS@orcl150> WHENEVER SQLERROR EXIT SQL.SQLCODE;
SYS@orcl150> REM
SYS@orcl150> VAR signature NUMBER;
SYS@orcl150> VAR signaturef NUMBER;
SYS@orcl150> REM
SYS@orcl150> DECLARE
  2  sql_txt CLOB;
  3  h       SYS.SQLPROF_ATTR;
  4  PROCEDURE wa (p_line IN VARCHAR2) IS
  5  BEGIN
  6  DBMS_LOB.WRITEAPPEND(sql_txt, LENGTH(p_line), p_line);
  7  END wa;
  8  BEGIN
  9  DBMS_LOB.CREATETEMPORARY(sql_txt, TRUE);
 10  DBMS_LOB.OPEN(sql_txt, DBMS_LOB.LOB_READWRITE);
 11  -- SQL Text pieces below do not have to be of same length.
 12  -- So if you edit SQL Text (i.e. removing temporary Hints),
 13  -- there is no need to edit or re-align unmodified pieces.
 14  wa(q'[select * from scott.test where object_id=10]');
 15  DBMS_LOB.CLOSE(sql_txt);
 16  h := SYS.SQLPROF_ATTR(
 17  q'[BEGIN_OUTLINE_DATA]',
 18  q'[IGNORE_OPTIM_EMBEDDED_HINTS]',
 19  q'[OPTIMIZER_FEATURES_ENABLE('11.2.0.4')]',
 20  q'[DB_VERSION('11.2.0.4')]',
 21  q'[ALL_ROWS]',
 22  q'[OUTLINE_LEAF(@"SEL$1")]',
 23  q'[FULL(@"SEL$1" "TEST"@"SEL$1")]',
 24  q'[END_OUTLINE_DATA]');
 25  :signature := DBMS_SQLTUNE.SQLTEXT_TO_SIGNATURE(sql_txt);
 26  :signaturef := DBMS_SQLTUNE.SQLTEXT_TO_SIGNATURE(sql_txt, TRUE);
 27  DBMS_SQLTUNE.IMPORT_SQL_PROFILE (
 28  sql_text    => sql_txt,
 29  profile     => h,
 30  name        => 'coe_cpk9jsg2qt52r_2317948335',
 31  description => 'coe cpk9jsg2qt52r 2317948335 '||:signature||' '||:signaturef||'',
 32  category    => 'DEFAULT',
 33  validate    => TRUE,
 34  replace     => TRUE,
 35  force_match => FALSE /* TRUE:FORCE (match even when different literals in SQL). FALSE:EXACT (similar to CURSOR_SHARING) */ );
 36  DBMS_LOB.FREETEMPORARY(sql_txt);
 37  END;
 38  /

PL/SQL procedure successfully completed.

SYS@orcl150> WHENEVER SQLERROR CONTINUE
SYS@orcl150> SET ECHO OFF;

            SIGNATURE
---------------------
 10910590721604799112


           SIGNATUREF
---------------------
 15966118871002195466


... manual custom SQL Profile has been created


COE_XFR_SQL_PROFILE_cpk9jsg2qt52r_2317948335 completed
2.2.2.8、查看最新SQL Profile
--查看产生的sql profile。
SYS@orcl150> set line 200
SYS@orcl150> r
  1  SELECT b.name, d.sql_text, extractvalue(value(h), '.') as hints
  2      FROM dba_sql_profiles d,
  3           SYS.SQLOBJ$DATA  a,
  4           SYS.SQLOBJ$ b,
  5           TABLE(XMLSEQUENCE(EXTRACT(XMLTYPE(A.COMP_DATA),'/outline_data/hint'))) h
  6    where a.signature=b.signature
  7      and a.category=b.category
  8      and a.obj_type=b.obj_type
  9      and a.plan_id=b.plan_id
 10      and a.signature=d.signature
 11*     and d.name='coe_cpk9jsg2qt52r_2317948335'

NAME                           SQL_TEXT                                           HINTS
------------------------------ -------------------------------------------------- --------------------------------------------------
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        BEGIN_OUTLINE_DATA
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        IGNORE_OPTIM_EMBEDDED_HINTS
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        OPTIMIZER_FEATURES_ENABLE('11.2.0.4')
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        DB_VERSION('11.2.0.4')
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        ALL_ROWS
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        OUTLINE_LEAF(@"SEL$1")
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        FULL(@"SEL$1" "TEST"@"SEL$1")
coe_cpk9jsg2qt52r_2317948335   select * from scott.test where object_id=10        END_OUTLINE_DATA

8 rows selected.
2.2.2.9、执行计划验证
--验证SQL Profile是否生效
SYS@orcl150> set autot traceonly explain stat
SYS@orcl150> select * from scott.test where object_id=10;

86420 rows selected.


Execution Plan
----------------------------------------------------------
Plan hash value: 1357081020

--------------------------------------------------------------------------
| Id  | Operation         | Name | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |      | 86404 |  8100K|   345   (1)| 00:00:05 |
|*  1 |  TABLE ACCESS FULL| TEST | 86404 |  8100K|   345   (1)| 00:00:05 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter("OBJECT_ID"=10)

Note
-----
   - SQL profile "coe_cpk9jsg2qt52r_2317948335" used for this statement


Statistics
----------------------------------------------------------
         28  recursive calls
          0  db block gets
       6925  consistent gets
         87  physical reads
          0  redo size
    4056770  bytes sent via SQL*Net to client
      63895  bytes received via SQL*Net from client
       5763  SQL*Net roundtrips to/from client
          3  sorts (memory)
          0  sorts (disk)
      86420  rows processed
说明:如上所示,原语句在不加hint的情况下也走全表扫描。
注意事项:
A、该测试只是为了演示通过coe_xfr_sql_profile.sql手动实现加hint,实际上述语句的处理最佳方法是重新收集scott.test的统计信息.
B、当一条sql既有sql profile又有stored outline时,优化器优先选择stored outline.
C、force_match参数,TRUE:FORCE (match even when different literals in SQL),FALSE:EXACT (similar to CURSOR_SHARING).
D、通过sql profile手动加hint的方法很简单,而为sql添加最合理的hint才是关键.
E、测试完后,可以通过exec dbms_sqltune.drop_sql_profile(name =>'coe_cpk9jsg2qt52r_3384190782' );删除该sql profile.
F、执行coe_xfr_sql_profile.sql脚本时需要用户对当前目录有生成文件的权限,建议当前目录是/tmp。
2.2.3、自动创建SQL Profile
2.2.3.1、相关概念
使用STA自动生成SQL Profile,STA会对语句进行分析,采用最优的优化策略,并给出优化后的执行计划,且可以按照STA给出的建议重写语句。但有时可能无法重写语句(比如在生产环境或语句在包中)。此时就可以利用sql profile,将优化策略存储在profile中,Oracle在构建SQL语句的执行计划时就不会使用已有的统计信息,而是使用profile的策略生成新的查询计划。
2.2.3.2、数据准备
SYS@orcl150> grant create any sql profile to leo;

Grant succeeded.

SYS@orcl150> grant drop any sql profile to leo;

Grant succeeded.

SYS@orcl150> grant alter any sql profile to leo;

Grant succeeded.

SYS@orcl150> conn leo/leo;
Connected.
LEO@orcl150> select * from v$version;

BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
PL/SQL Release 11.2.0.4.0 - Production
CORE    11.2.0.4.0      Production
TNS for Linux: Version 11.2.0.4.0 - Production
NLSRTL Version 11.2.0.4.0 – Production
LEO@orcl150> create table tb_leo_20230305_01 as select * from dba_objects;

Table created.

LEO@orcl150> create index tb_leo_20230305_01_idx on tb_leo_20230305_01(object_id);

Index created.

LEO@orcl150> exec dbms_stats.gather_table_stats('leo','TB_LEO_20230305_01',cascade=>true,degree=>4);

PL/SQL procedure successfully completed.
LEO@orcl150> set autot on
LEO@orcl150> select /*+no_index(tb_leo_20230305_01 tb_leo_20230305_01_idx)*/ count(*) from tb_leo_20230305_01 where object_id=100;

  COUNT(*)
----------
         1


Execution Plan
----------------------------------------------------------
Plan hash value: 65431150

-----------------------------------------------------------------------------------------
| Id  | Operation          | Name               | Rows  | Bytes | Cost (%CPU)| Time     |
-----------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |                    |     1 |     5 |   345   (1)| 00:00:05 |
|   1 |  SORT AGGREGATE    |                    |     1 |     5 |            |          |
|*  2 |   TABLE ACCESS FULL| TB_LEO_20230305_01 |     1 |     5 |   345   (1)| 00:00:05 |
-----------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - filter("OBJECT_ID"=100)


Statistics
----------------------------------------------------------
          1  recursive calls
          0  db block gets
       1236  consistent gets
       1234  physical reads
          0  redo size
        526  bytes sent via SQL*Net to client
        524  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
1	rows processed
LEO@orcl150> set autot off
LEO@orcl150> col sql_text for a100
LEO@orcl150> select v.sql_id,v.sql_text from v$sql v where v.sql_text like '%no_index(tb_leo_20230305_01%' and v.sql_text not like '%v$sql%';

SQL_ID        SQL_TEXT
------------- ----------------------------------------------------------------------------------------------------
bvx7dwxdhc2j5 EXPLAIN PLAN SET STATEMENT_ID='PLUS340381' FOR select /*+no_index(tb_leo_20230305_01 tb_leo_20230305
              _01_idx)*/ count(*) from tb_leo_20230305_01 where object_id=100

bkms28pu7749m select /*+no_index(tb_leo_20230305_01 tb_leo_20230305_01_idx)*/ count(*) from tb_leo_20230305_01 whe
              re object_id=100
2.2.3.3、创建优化任务
方法1:
LEO@orcl150> declare
  2    my_task_name varchar2(30);
  3    my_sqltext   clob;
  4  begin
  5    my_sqltext   := 'select /*+no_index(tb_leo_20230305_01 tb_leo_20230305_01_idx)*/ count(*) from tb_leo_20230305_01 where object_id=100';
  6    my_task_name := dbms_sqltune.create_tuning_task(sql_text    => my_sqltext,
  7                                                    user_name   => 'LEO',
  8                                                    scope       => 'COMPREHENSIVE',
  9                                                    time_limit  => 60,
 10                                                    task_name   => 'sql_profile_test',
 11                                                    description => 'Task to tune a query on a specified table');
 12    dbms_sqltune.execute_tuning_task(task_name => 'sql_profile_test');
 13  end;
 14  /

PL/SQL procedure successfully completed.
方法2:
使用sqlid来生成优化任务,如下:
LEO@orcl150> declare
  2    a_tuning_task varchar2(30);
  3  begin
  4    a_tuning_task := dbms_sqltune.create_tuning_task(sql_id    => 'bkms28pu7749m',
  5                                             task_name => 'sql_profile_test_sqlid');
  6    dbms_sqltune.execute_tuning_task(a_tuning_task);
  7  end;
  8  /

PL/SQL procedure successfully completed.
2.2.3.4、查看优化建议
LEO@orcl150> set autot off
LEO@orcl150> set long 10000
LEO@orcl150> set longchunksize 1000
LEO@orcl150> set linesize 100
LEO@orcl150> select dbms_sqltune.report_tuning_task('sql_profile_test') from dual;

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
GENERAL INFORMATION SECTION
-------------------------------------------------------------------------------
Tuning Task Name   : sql_profile_test
Tuning Task Owner  : LEO
Workload Type      : Single SQL Statement
Scope              : COMPREHENSIVE
Time Limit(seconds): 60
Completion Status  : COMPLETED
Started at         : 03/06/2023 10:24:30
Completed at       : 03/06/2023 10:24:30


DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
-------------------------------------------------------------------------------
Schema Name: LEO
SQL ID     : 3ghugnm7mr3v0
SQL Text   : select /*+no_index(tb_leo_20230305_01 tb_leo_20230305_01_idx)*/
             count(*) from tb_leo_20230305_01 where object_id=100

-------------------------------------------------------------------------------
FINDINGS SECTION (1 finding)
-------------------------------------------------------------------------------

1- SQL Profile Finding (see explain plans section below)

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
--------------------------------------------------------
  A potentially better execution plan was found for this statement.

  Recommendation (estimated benefit: 99.83%)
  ------------------------------------------
  - Consider accepting the recommended SQL profile.
    execute dbms_sqltune.accept_sql_profile(task_name => 'sql_profile_test',
            task_owner => 'LEO', replace => TRUE);

  Validation results
  ------------------

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
  The SQL profile was tested by executing both its plan and the original plan
  and measuring their respective execution statistics. A plan may have been
  only partially executed if the other could be run to completion in less time.

                           Original Plan  With SQL Profile  % Improved
                           -------------  ----------------  ----------
  Completion Status:            COMPLETE          COMPLETE
  Elapsed Time (s):             .003286           .000017      99.48 %
  CPU Time (s):                 .003286           .000017      99.48 %
  User I/O Time (s):            .000116                 0        100 %
  Buffer Gets:                     1238                 2      99.83 %

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
  Physical Read Requests:            15                 0        100 %
  Physical Write Requests:            0                 0
  Physical Read Bytes:           376832                 0        100 %
  Physical Write Bytes:               0                 0
  Rows Processed:                     1                 1
  Fetches:                            1                 1
  Executions:                         1                 1

  Notes
  -----
  1. Statistics for the original plan were averaged over 10 executions.

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
  2. Statistics for the SQL profile plan were averaged over 10 executions.

-------------------------------------------------------------------------------
EXPLAIN PLANS SECTION
-------------------------------------------------------------------------------

1- Original With Adjusted Cost
------------------------------
Plan hash value: 65431150

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

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
| Id  | Operation          | Name               | Rows  | Bytes | Cost (%CPU)| Time     |
-----------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |                    |     1 |     5 |   345   (1)| 00:00:05 |
|   1 |  SORT AGGREGATE    |                    |     1 |     5 |            |          |
|*  2 |   TABLE ACCESS FULL| TB_LEO_20230305_01 |     1 |     5 |   345   (1)| 00:00:05 |
-----------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - filter("OBJECT_ID"=100)

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------

2- Using SQL Profile
--------------------
Plan hash value: 1195562460

--------------------------------------------------------------------------------------------
| Id  | Operation         | Name                   | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                        |     1 |     5 |     1   (0)| 00:00:01 |
|   1 |  SORT AGGREGATE   |                        |     1 |     5 |            |          |
|*  2 |   INDEX RANGE SCAN| TB_LEO_20230305_01_IDX |     1 |     5 |     1   (0)| 00:00:01 |

DBMS_SQLTUNE.REPORT_TUNING_TASK('SQL_PROFILE_TEST')
----------------------------------------------------------------------------------------------------
--------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("OBJECT_ID"=100)

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

说明:可以看到优化器给出性能更高的执行计划。
2.2.3.5、接受profile
--决定接受该建议,使用最新执行计划,且不重写SQL语句.
LEO@orcl150> execute dbms_sqltune.accept_sql_profile(task_name => 'sql_profile_test',task_owner => 'LEO',replace => TRUE);

PL/SQL procedure successfully completed.
注意:task_name需与前面创建优化任务时的task_name对应.
说明:采用包DBMS_SQLTUNE的函数:ACCEPT_SQL_PROFILE,其中参数task_name即创建的优化建议任务的名称,可以是任意合法名称。该函数还有其他参数,下面是该函数原型:
DBMS_SQLTUNE.ACCEPT_SQL_PROFILE (
   task_name    IN  VARCHAR2,
   object_id    IN  NUMBER   := NULL,
   name         IN  VARCHAR2 := NULL,
   description  IN  VARCHAR2 := NULL,
   category     IN  VARCHAR2 := NULL;
   task_owner   IN VARCHAR2  := NULL,
   replace      IN BOOLEAN   := FALSE,
   force_match  IN BOOLEAN   := FALSE)
RETURN VARCHAR2;
Description是profile的描述信息;task_owner是优化建议任务的所有者;replace为TRUE时,如果profile已经存在,就代替它;force_match为TURE时,表示与语句强制匹配,即强制使用绑定变量,和系统参数cursor_sharing设置为FORCE时类似。为FALSE时,与cursor_sharing设置为EXACT时类似,即完全匹配。
特别是category参数,可以通过设置该参数制定特定会话使用该profile。10g中每个会话都有一个新参数SQLTUNE_CATEGORY,默认值是DEFAULT。而在调用该函数时,如果没有指定此参数,其值也是DEFAULT,而如果给这个profile指定了一个其它的CATEGORY值,如FOR_TUNING,那么只有会话参数SQLTUNE_CATEGORY也为FOR_TUNING时,才会使用这个porfile。为什么说这个参数很有用呢?试想这样的环境:在一个生产系统上利用STA调优一条语句,STA已经给出了优化建议,但是又不敢贸然实施它给出的建议(毕竟它只是机器嘛,不能完全信任),你就可以创建一个有特殊CATEGORY的profile,然后在你自己的会话中制定SQLTUNE_CATEGORY为这个特殊的CATEGORY,那就既可以看优化建议的实际效果又不影响生产环境。
2.2.3.6、查看创建的profile
--通过视图DBA_SQL_PROFILES查看已经创建的profile。
LEO @orcl150> set autot off
LEO @orcl150> col SQL_TEXT for a35
LEO @orcl150> r
  1  SELECT e.task_name, b.name, d.sql_text, extractvalue(value(h), '.') as hints
  2      FROM dba_sql_profiles d,
  3          dba_advisor_tasks e,
  4           SYS.SQLOBJ$DATA  a,
  5           SYS.SQLOBJ$ b,
  6           TABLE(XMLSEQUENCE(EXTRACT(XMLTYPE(A.COMP_DATA),'/outline_data/hint'))) h
  7    where a.signature=b.signature
  8      and a.category=b.category
  9      and a.obj_type=b.obj_type
 10      and a.plan_id=b.plan_id
 11      and a.signature=d.signature
 12     and d.task_id=e.task_id
 13      and d.name='SYS_SQLPROF_0186b6080db30001'
 14*

TASK_NAME         NAME                           SQL_TEXT                            HINTS
----------------- ------------------------------ ----------------------------------- -----------------------------------
sql_profile_test  SYS_SQLPROF_0186b6080db30001   select /*+no_index(tb_leo_2023030 OPTIMIZER_FEATURES_ENABLE(default)
                                                 5_01 tb_leo_20230305_01_idx)*/ co
                                                 unt(*) from tb_leo_20230305_01 wh
                                                 ere object_id=100

sql_profile_test  SYS_SQLPROF_0186b6080db30001   select /*+no_index(tb_leo_2023030 IGNORE_OPTIM_EMBEDDED_HINTS
                                                 5_01 tb_leo_20230305_01_idx)*/ co
                                                 unt(*) from tb_leo_20230305_01 wh
                                                 ere object_id=100

注意:SYS_SQLPROF_0186b6080db30001来自后文profile效果查看时显示的参数.
2.2.3.7、查看profile效果
2.2.3.7.1、sqlplus查看
LEO@orcl150> set autot on
LEO@orcl150> select /*+no_index(tb_leo_20230305_01 tb_leo_20230305_01_idx)*/ count(*) from tb_leo_20230305_01 where object_id=100;

  COUNT(*)
----------
         1


Execution Plan
----------------------------------------------------------
Plan hash value: 1195562460

--------------------------------------------------------------------------------------------
| Id  | Operation         | Name                   | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |                        |     1 |     5 |     1   (0)| 00:00:01 |
|   1 |  SORT AGGREGATE   |                        |     1 |     5 |            |          |
|*  2 |   INDEX RANGE SCAN| TB_LEO_20230305_01_IDX |     1 |     5 |     1   (0)| 00:00:01 |
--------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("OBJECT_ID"=100)

Note
-----
   - SQL profile "SYS_SQLPROF_0186b6080db30001" used for this statement


Statistics
----------------------------------------------------------
          8  recursive calls
          0  db block gets
          7  consistent gets
          1  physical reads
          0  redo size
        526  bytes sent via SQL*Net to client
        524  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
          0  sorts (memory)
          0  sorts (disk)
          1  rows processed
说明:该语句采用profile的数据,执行计划中的附加信息表明该语句采用’SYS_SQLPROF_0186b6080db30001’ profile,而不是根据对象上的统计数据来生成的查询计划。
上述方法主要依赖sql tuning advisor,如果它无法生成想要的执行计划。还可以通过sql profile手动把hint加进去。复杂SQL的hint可以采用脚本coe_xfr_sql_profile.sql来产生原语句的outline data和加hint语句的outline data,然后替换对应的SYS.SQLPROF_ATTR,最后执行生成的sql。
2.2.3.7.2、PL/SQL查看
使用PLSQL DEVELOPER 14查看执行计划,如下所示:

oracle固定执行计划_执行计划

参考文档:https://blog.51cto.com/lhrbest/3246884

精彩评论(0)

0 0 举报