Oracle 11g为SQL生成合适执行计划并固定
Posted ~沉%淀~
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Oracle 11g为SQL生成合适执行计划并固定相关的知识,希望对你有一定的参考价值。
------------恢复内容开始------------
1.查询当前SQL执行计划并固定
(1)查询指定SQL SQL_ID
SQL> select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text like \'%fundinfo%\';
SQL_ID SQL_TEXT VERSION_COUNT PARSE_CALLS
------------- -------------------------------------------------------------------------------- ------------- -----------
6v57a84f59hqk select count(planid) from fundinfo where planid < 99000014000800382760 1 1
adz60h2w2js6p select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text l 1 1
(2)查询SQL_ID对应执行计划
# 使用下面两种查询一种查询出对应执行计划即可
SQL> select * from table(dbms_xplan.display_cursor(\'6v57a84f59hqk\'));
PLAN_TABLE_OUTPUT
--------------------------------------------------------------------------------
SQL_ID 6v57a84f59hqk, child number 0
-------------------------------------
select count(planid) from fundinfo where planid < 99000014000800382760
Plan hash value: 913026973
---------------------------------------
| Id | Operation | Name |
---------------------------------------
| 0 | SELECT STATEMENT | |
| 1 | SORT AGGREGATE | |
|* 2 | TABLE ACCESS FULL| FUNDINFO |
---------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
2 - filter(TO_NUMBER("PLANID")<99000014000800382760)
Note
PLAN_TABLE_OUTPUT
--------------------------------------------------------------------------------
-----
- rule based optimizer used (consider using cbo)
23 rows selected
SQL> SELECT hash_value,sql_id,plan_hash_value,operation,options FROM v$sql_plan WHERE sql_id=\'6v57a84f59hqk\';
HASH_VALUE SQL_ID PLAN_HASH_VALUE OPERATION OPTIONS
---------- ------------- --------------- ------------------------------ ------------------------------
475316946 6v57a84f59hqk 913026973 SELECT STATEMENT
475316946 6v57a84f59hqk 913026973 SORT AGGREGATE
475316946 6v57a84f59hqk 913026973 TABLE ACCESS FULL
(3)固定执行计划
SQL> var test number
SQL> begin
2 :test := dbms_spm.load_plans_from_cursor_cache(sql_id=>\'6v57a84f59hqk\', plan_hash_value=>913026973);
3 end;
4 /
PL/SQL procedure successfully completed
SQL> select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf36b969528 MANUAL-LOAD
2.使用HINT生成指定的执行计划
(1)使用/*+ INDEX(table_name index_name)+*/生成执行计划
HR@honor1 > select /*+ INDEX(FUNDINFO FUNDINFO_PLANID)+*/ count(planid) from fundinfo where planid < 99000014000800382760;
COUNT(PLANID)
----------------------------------------
337473
(2)查询生成的执行计划
SQL> select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text like \'%fundinfo%\';
SQL_ID SQL_TEXT VERSION_COUNT PARSE_CALLS
------------- -------------------------------------------------------------------------------- ------------- -----------
gw6gd2ggk0h9m select /*+ INDEX(FUNDINFO FUNDINFO_PLANID)+*/ count(planid) from fundinfo where 1 1
adz60h2w2js6p select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text l 1 3
SQL> SELECT hash_value,sql_id,plan_hash_value,operation,options FROM v$sql_plan WHERE sql_id=\'gw6gd2ggk0h9m\';
HASH_VALUE SQL_ID PLAN_HASH_VALUE OPERATION OPTIONS
---------- ------------- --------------- ------------------------------ ------------------------------
3743433011 gw6gd2ggk0h9m 2614089935 SELECT STATEMENT
3743433011 gw6gd2ggk0h9m 2614089935 SORT AGGREGATE
3743433011 gw6gd2ggk0h9m 2614089935 INDEX FULL SCAN
3.使用dbms_spm.load_plans_from_cursor_cache函数固定执行计划
# 说明
begin
:test := dbms_spm.load_plans_from_cursor_cache(
sql_id => \'加入HINT后的SQL_ID\',
plan_hash_value => 加入HINT后生成的plan_hash_value,
sql_handle => \'旧的执行计划固定后dba_sql_plan_baselines中sql_handle\');
end;
/
# 执行
SQL> begin
2 :test := dbms_spm.load_plans_from_cursor_cache(
3 sql_id => \'gw6gd2ggk0h9m\',
4 plan_hash_value => 2614089935,
5 sql_handle => \'SQL_458606bd5b57f1c3\');
6 end;
7 /
PL/SQL procedure successfully completed
# 查询基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf3686e388b MANUAL-LOAD
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf36b969528 MANUAL-LOAD
4.删除最开始固定的旧基线,使新的执行计划的新基线生效
SQL> var test number;
SQL> begin
2 :test := dbms_spm.drop_sql_plan_baseline(sql_handle=>\'SQL_458606bd5b57f1c3\', plan_name=>\'SQL_PLAN_4b1h6rpdpgwf36b969528\');
3 end;
4 /
PL/SQL procedure successfully completed
# 查询基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf3686e388b MANUAL-LOAD
5.测试新基线
HR@honor1 > explain plan for select count(planid) from fundinfo where planid < 99000014000800382760;
Explained.
HR@honor1 > select * from table(dbms_xplan.display);
PLAN_TABLE_OUTPUT
------------------------------------------------------------------------------------------------------------------------
Plan hash value: 2614089935
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 22 | 27 (4)| 00:00:01 |
| 1 | SORT AGGREGATE | | 1 | 22 | | |
|* 2 | INDEX FULL SCAN| FUNDINFO_PLANID | 18023 | 387K| 27 (4)| 00:00:01 |
------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
2 - filter(TO_NUMBER("PLANID")<99000014000800382760)
Note
-----
- SQL plan baseline "SQL_PLAN_4b1h6rpdpgwf3686e388b" used for this statement
18 rows selected.
# 可以看到走了索引全扫描,新基线已经生效。
6.删除基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf37d78711b AUTO-CAPTURE
var test number;
begin
:test := dbms_spm.drop_sql_plan_baseline(sql_handle=>\'SQL_458606bd5b57f1c3\', plan_name=>\'SQL_PLAN_4b1h6rpdpgwf37d78711b\');
end;
/
PL/SQL procedure successfully completed.
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
no rows selected
参考文档:
https://www.cnblogs.com/ddzj01/p/11377049.html
------------恢复内容结束------------
------------恢复内容开始------------
1.查询当前SQL执行计划并固定
(1)查询指定SQL SQL_ID
SQL> select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text like \'%fundinfo%\';
SQL_ID SQL_TEXT VERSION_COUNT PARSE_CALLS
------------- -------------------------------------------------------------------------------- ------------- -----------
6v57a84f59hqk select count(planid) from fundinfo where planid < 99000014000800382760 1 1
adz60h2w2js6p select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text l 1 1
(2)查询SQL_ID对应执行计划
# 使用下面两种查询一种查询出对应执行计划即可
SQL> select * from table(dbms_xplan.display_cursor(\'6v57a84f59hqk\'));
PLAN_TABLE_OUTPUT
--------------------------------------------------------------------------------
SQL_ID 6v57a84f59hqk, child number 0
-------------------------------------
select count(planid) from fundinfo where planid < 99000014000800382760
Plan hash value: 913026973
---------------------------------------
| Id | Operation | Name |
---------------------------------------
| 0 | SELECT STATEMENT | |
| 1 | SORT AGGREGATE | |
|* 2 | TABLE ACCESS FULL| FUNDINFO |
---------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
2 - filter(TO_NUMBER("PLANID")<99000014000800382760)
Note
PLAN_TABLE_OUTPUT
--------------------------------------------------------------------------------
-----
- rule based optimizer used (consider using cbo)
23 rows selected
SQL> SELECT hash_value,sql_id,plan_hash_value,operation,options FROM v$sql_plan WHERE sql_id=\'6v57a84f59hqk\';
HASH_VALUE SQL_ID PLAN_HASH_VALUE OPERATION OPTIONS
---------- ------------- --------------- ------------------------------ ------------------------------
475316946 6v57a84f59hqk 913026973 SELECT STATEMENT
475316946 6v57a84f59hqk 913026973 SORT AGGREGATE
475316946 6v57a84f59hqk 913026973 TABLE ACCESS FULL
(3)固定执行计划
SQL> var test number
SQL> begin
2 :test := dbms_spm.load_plans_from_cursor_cache(sql_id=>\'6v57a84f59hqk\', plan_hash_value=>913026973);
3 end;
4 /
PL/SQL procedure successfully completed
SQL> select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf36b969528 MANUAL-LOAD
2.使用HINT生成指定的执行计划
(1)使用/*+ INDEX(table_name index_name)+*/生成执行计划
HR@honor1 > select /*+ INDEX(FUNDINFO FUNDINFO_PLANID)+*/ count(planid) from fundinfo where planid < 99000014000800382760;
COUNT(PLANID)
----------------------------------------
337473
(2)查询生成的执行计划
SQL> select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text like \'%fundinfo%\';
SQL_ID SQL_TEXT VERSION_COUNT PARSE_CALLS
------------- -------------------------------------------------------------------------------- ------------- -----------
gw6gd2ggk0h9m select /*+ INDEX(FUNDINFO FUNDINFO_PLANID)+*/ count(planid) from fundinfo where 1 1
adz60h2w2js6p select sql_id,sql_text,version_count,parse_calls from v$sqlarea where sql_text l 1 3
SQL> SELECT hash_value,sql_id,plan_hash_value,operation,options FROM v$sql_plan WHERE sql_id=\'gw6gd2ggk0h9m\';
HASH_VALUE SQL_ID PLAN_HASH_VALUE OPERATION OPTIONS
---------- ------------- --------------- ------------------------------ ------------------------------
3743433011 gw6gd2ggk0h9m 2614089935 SELECT STATEMENT
3743433011 gw6gd2ggk0h9m 2614089935 SORT AGGREGATE
3743433011 gw6gd2ggk0h9m 2614089935 INDEX FULL SCAN
3.使用dbms_spm.load_plans_from_cursor_cache函数固定执行计划
# 说明
begin
:test := dbms_spm.load_plans_from_cursor_cache(
sql_id => \'加入HINT后的SQL_ID\',
plan_hash_value => 加入HINT后生成的plan_hash_value,
sql_handle => \'旧的执行计划固定后dba_sql_plan_baselines中sql_handle\');
end;
/
# 执行
SQL> begin
2 :test := dbms_spm.load_plans_from_cursor_cache(
3 sql_id => \'gw6gd2ggk0h9m\',
4 plan_hash_value => 2614089935,
5 sql_handle => \'SQL_458606bd5b57f1c3\');
6 end;
7 /
PL/SQL procedure successfully completed
# 查询基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf3686e388b MANUAL-LOAD
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf36b969528 MANUAL-LOAD
4.删除最开始固定的旧基线,使新的执行计划的新基线生效
SQL> var test number;
SQL> begin
2 :test := dbms_spm.drop_sql_plan_baseline(sql_handle=>\'SQL_458606bd5b57f1c3\', plan_name=>\'SQL_PLAN_4b1h6rpdpgwf36b969528\');
3 end;
4 /
PL/SQL procedure successfully completed
# 查询基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf3686e388b MANUAL-LOAD
5.测试新基线
HR@honor1 > explain plan for select count(planid) from fundinfo where planid < 99000014000800382760;
Explained.
HR@honor1 > select * from table(dbms_xplan.display);
PLAN_TABLE_OUTPUT
------------------------------------------------------------------------------------------------------------------------
Plan hash value: 2614089935
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 22 | 27 (4)| 00:00:01 |
| 1 | SORT AGGREGATE | | 1 | 22 | | |
|* 2 | INDEX FULL SCAN| FUNDINFO_PLANID | 18023 | 387K| 27 (4)| 00:00:01 |
------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
2 - filter(TO_NUMBER("PLANID")<99000014000800382760)
Note
-----
- SQL plan baseline "SQL_PLAN_4b1h6rpdpgwf3686e388b" used for this statement
18 rows selected.
# 可以看到走了索引全扫描,新基线已经生效。
6.删除基线
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
SQL_HANDLE PLAN_NAME ORIGIN
------------------------------ ------------------------------ --------------
SQL_458606bd5b57f1c3 SQL_PLAN_4b1h6rpdpgwf37d78711b AUTO-CAPTURE
var test number;
begin
:test := dbms_spm.drop_sql_plan_baseline(sql_handle=>\'SQL_458606bd5b57f1c3\', plan_name=>\'SQL_PLAN_4b1h6rpdpgwf37d78711b\');
end;
/
PL/SQL procedure successfully completed.
SYS@honor1 > select sql_handle, plan_name, origin from dba_sql_plan_baselines;
no rows selected
参考文档:
https://www.cnblogs.com/ddzj01/p/11377049.html
------------恢复内容结束------------
以上是关于Oracle 11g为SQL生成合适执行计划并固定的主要内容,如果未能解决你的问题,请参考以下文章
ORACLE 11G SPM(SQL PLAN manager)浅析