我们都希望对于所有在Oracle数据库中执行的sql,CBO都能产生出正确的执行计划,但实际情况却并非如此,由于各种各样的原因(比如目标sql所涉及的对象的统计信息的不准确,或者CBO内部一些成本计算公式的先天缺陷等),导致有时CBO产生效率不高、甚至是错误的执行计划。特别是CBO对目标sql所产生的初始执行计划是正确的,后来由于某种原因(比如统计信息的变更等)而导致CBO重新对其产生了错误的执行计划,这种执行计划的改变往往会导致目标sql执行时间呈数量级的递增,而且常常会让我们很困惑:这个sql原先跑得好好的,为什么突然就慢得让人无法接受?其实这种sql执行效率突然衰减往往是因为目标sql执行计划的改变。
我们当然希望这样的改变永远不要发生,即在Oracle数据库中跑的所有sql都能有正确的、稳定的执行计划,但实际上在Oracle 11g的SPM(sql Plan Management)出现之前,这一点是很难做到的。那么现在退而求其次,如果已经出现了执行坟墓的变更,即CBO已经产生了错误的执行计划,我们应该怎么纠正呢?
我种情况下,我们通常会重新收集一下统计信息或者修改目标sql(比如在目标sql中加入Hint等)以纠正错误的执行计划。但有时候重新收集统计信息并不能解决问题,更糟糕的是,很多情况下是没有办法修改目标sql的sql文本的(比如第三方开发的系统,修改不了源码,或者目标sql是前台框架动态生成的等等),那么这种情况下我们该怎么办呢?
在Oracle 10g/11g及其以后的版本中,我们可以使用sql Profile或SPM(sql Plan Management)来解决上述执行计划变更的问题,用它们来调整、稳定目标sql的执行计划。
本文介绍使用sql Profile来稳定执行计划:
Oracle 10g中的sql Profile(直译为“sql概要”)可以说是Oracle 9i中的Stored Outline(直译为“存储概要”)的进化。Stored Outline能够实现的功能sql Profile也完全能够实现。
与Stored Outline相比,sql Profile具备如下优点:
使用sql Profile可以很容易实现如下两个目的:
sql Profile有两种类型:一种是Automatic类型,另一种是Manual类型。下面分别介绍这两种类型:
1. Automatic类型的sql Profile
Automatic类型的sql Profile其实就是针对目标sql的一些额外的调整信息,这些信息存储在数据字典里。当有了Automatic类型的sql Profile后,Oracle在产生执行计划时就会根据它对目标sql所涉及的统计信息等内容做相应的调整,因而能够在一定程度上避免产生错误的执行计划。你不用担心Automatic类型的sql Profile的准确性,因为Oracle会使用类型于动态采用技术那样的手段来保证这些额外调整信息相对准确。
Automatic类型的sql Profile不会像Stored Outline那样锁定目标sql的执行计划,因为Automatic类型的sql Profile的本质就是针对目标sql的一些额外的调整信息,这些额外的调整信息需要与原目标sql的相关统计信息等内容一起作用才能得到新的执行计划,即原始sql的统计信息等内容一旦发生变化,即使原有Automatic类型的sql Profile并没有改变,该sql的执行也可能会发生变化。从这个意义上讲,Automatic类型的sql Profile并不能完全起到稳定目标sql的执行计划的作用,虽然它确实可以用来调整执行计划。
看一个在不更改目标sql的sql文本的情况下使用Automatic类型的sql Profile来调整执行计划的实例:
创建测试表及相关操作:
zx@MYDB>createtablet1(nnumber); Tablecreated. zx@MYDB>declare 2begin 3foriin1..10000loop 4insertintot1values(i); 5endloop; 6commit; 7end; 8/ PL/sqlproceduresuccessfullycompleted. zx@MYDB>selectcount(*)fromt1; COUNT(*) ---------- 10000 zx@MYDB>createindexidx_t1ont1(n); Indexcreated. zx@MYDB>execdbms_stats.gather_table_stats(ownname=>USER,tabname=>'T1',method_opt=>'forallcolumnssize1',cascade=>true); PL/sqlproceduresuccessfullycompleted. zx@MYDB>select/*+no_index(t1idx_t1)*/*fromt1wheren=1; N ---------- 1
从上述显示内容可以看出,目标sql走的是对表T1的全表扫描(Table Access Full),这个执行计划显然是错误,这里正确的执行坟墓应该是走索引IDX_T1的索引范围扫描(Index Range Scan)。下面使用sql Tuning Advisor对这条sql生成Automatic类型的sql Profile。
a.先创建一个名为my_sql_tuning_task_2的自动调整任务:
zx@MYDB>declare 2my_task_namevarchar2(30); 3my_sqltextclob; 4begin 5my_sqltext:='select/*+no_index(t1idx_t1)*/*fromt1wheren=1'; 6my_task_name:=dbms_sqltune.create_tuning_task( 7sql_text=>my_sqltext,8user_name=>USER,9scope=>'COMPREHENSIVE',10time_limit=>60,11task_name=>'my_sql_tuning_task_1',12description=>'Tasktotuneaqueryontablet1'); 13end; 14/ PL/sqlproceduresuccessfullycompleted. zx@MYDB>selecttask_name,status,execution_start,execution_endfromuser_advisor_log; TASK_NAMESTATUSEXECUTION_STARTEXECUTION_END ----------------------------------------------------------------------------------------------------- my_sql_tuning_task_1INITIAL
注:创建任务时可以使用sql来创建,可以适用于sql文本长的情况。详情参考官方文档。
b.执行上述自动调整任务
zx@MYDB>begin 2dbms_sqltune.execute_tuning_task(task_name=>'my_sql_tuning_task_1'); 3end; 4/ zx@MYDB>zx@MYDB>selecttask_name,execution_endfromuser_advisor_log; TASK_NAMESTATUSEXECUTION_STARTEXECUTION_END ----------------------------------------------------------------------------------------------------- my_sql_tuning_task_1COMPLETED2017-02-2810:59:432017-02-2810:59:44 PL/sqlproceduresuccessfullycompleted.
c.查看上述自动任务的调整结果
zx@MYDB>setlong9000 zx@MYDB>setlongchunksize1000 zx@MYDB>setlinesize800 zx@MYDB>selectdbms_sqltune.report_tuning_task('my_sql_tuning_task_1')fromdual;
从上述调整结果可以看到,Oracle现在告诉我们:它已经为目标sql找到了更好的执行计划,并且已经创建了针对该sql的Automatic类型的sql Profile。如果我们使用accecp_sql_profile接受了这个sql Profile,则目标sql的响应时间将会有86.24%的提升,逻辑读将会有95%的提升,并且接受了该sql Profile后目标sql的执行计划将会由原来的全表扫描变为索引范围扫描。
上面Automatic类型的sql Profile所产生的调整结果就是我们想要的,所以现在只需按Oracle的提示接受这个sql Profile即可:
zx@MYDB>executedbms_sqltune.accept_sql_profile(task_name=>'my_sql_tuning_task_1',task_owner=>'ZX',replace=>TRUE,force_match=>true); PL/sqlproceduresuccessfullycompleted.
接受此sql Profile后我们来看一下效果,再次执行目标sql:
zx@MYDB>select/*+no_index(t1idx_t1)*/*fromt1wheren=1; N ---------- 1
注意到Note部分有这样的内容“sql profile SYS_sqlPROF_015a82b353490000 used for this statement”这说明我们刚才接受的sql Profile已经起了作用,该sql Profile的名字为SYS_sqlPROF_015a82b353490000。从执行计划中也可以看到,执行计划确实已经改变了。
另外,DBMS_sqlTUNE.ACCEPT_sql_PROFILE的输入参数force_match的默认值为FALSE,表示只有在sql文本完全匹配的情况下才会应用sql Profile,这种情况下只要目标sql的sql文本发生一点变动,原有的sql Profile将会失去作用,如果设置为TRUE,即使sql有变动sql Profile也会强制生效。
zx@MYDB>execdbms_sqltune.drop_sql_profile('SYS_sqlPROF_015a82b353490000'); PL/sqlproceduresuccessfullycompleted.
2. Manual类型的sql Profile
Manual类型的sql Profile本质上就是一堆Hint的组合,这一堆Hint的组合实际上来源于执行计划中的Outline Data部分的Hint组合。Manual类型的sql Profile同样可以在不更改目标sql的sql文本的情况下,调整其执行计划,而且更为重要的是,Manual类型的sql Profile可以起到很好稳定目标sql的执行计划的作用,这一点是Automatic类型的sql Profile所不具备的。
看一个使用Manual类型的sql Profile实例固定执行计划的实例,使用上面的t1表,删除上面的sql Profile,再次执行sql
zx@MYDB>select/*+no_index(t1idx_t1)*/*fromt1wheren=1; N ---------- 1
从上述输出可以看出执行计划仍然走全表扫描。
现在来创建Manual类型的sql Profile。这里使用了MOS上的一个脚本coe_xfr_sql_profile.sql。这个脚本用于从Shared Pool、AWR Repository中指定sql的指定执行计划的Outline Data部分的Hint组合,来创建Manual类型的sql Profile。
使用coe_xfr_sql_profile.sql脚本的步骤为
针对目标sql使用coe_xfr_sql_profile.sql产生能生成其Manual类型的sql Profile的脚本A。
改写目标sql的文本,在其中使用合适的Hint,直到加入Hint后的sql能走出我们想要的执行计划。然后对加入合适Hint后的sql使用脚本coe_xfr_sql_profile.sql,产生能生成其Manual类型的sql Profile的脚本B。
用脚本B中的Outline Data部分的Hint组合替换掉脚本A的Outline Data部分的Hint组合。
现在改写上面的sql,强制走索引:
zx@MYDB>select/*+index(t1idx_t1)*/*fromt1wheren=1; N ---------- 1
从执行计划中可以看出sql Id和对应的Plan hash value。
全表扫描的sql Id:6chcc0pvvhqqm Plan hash value:3617692013
索引扫描的sql Id:2ufquy7xs5nm5Plan hash value:1369807930
a. 先使用coe_xfr_sql_profile.sql生成全表扫描sql对应的脚本
zx@MYDB>@scripts/coe_xfr_sql_profile.sql Parameter1: sql_ID(required) Entervaluefor1:6chcc0pvvhqqm PLAN_HASH_VALUEAVG_ET_SECS -------------------------- 3617692013.002 Parameter2: PLAN_HASH_VALUE(required) Entervaluefor2:3617692013 Valuespassedtocoe_xfr_sql_profile: ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ sql_ID :"6chcc0pvvhqqm" PLAN_HASH_VALUE:"3617692013" sql>BEGIN 2IF:sql_textISNULLTHEN 3 RAISE_APPLICATION_ERROR(-20100,'sql_TEXTforsql_ID&&sql_id.wasnotfoundinmemory(gv$sqltext_with_newlines)orAWR(dba_hist_sqltext).'); 4ENDIF; 5END; 6/ sql>SETTERMOFF; sql>BEGIN 2IF:other_xmlISNULLTHEN 3 RAISE_APPLICATION_ERROR(-20101,'PLANforsql_ID&&sql_id.andPHV&&plan_hash_value.wasnotfoundinmemory(gv$sql_plan)orAWR(dba_hist_sql_plan).'); 4ENDIF; 5END; 6/ sql>SETTERMOFF; Executecoe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql onTARGETsysteminordertocreateacustomsqlProfile withplan3617692013linkedtoadjustedsql_text. COE_XFR_sql_PROFILEcompleted.
从输出可以看出,生成一个名为coe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql的脚本。
b.用coe_xfr_sql_profile.sql生成索引扫描sql对应的脚本
sql>@scripts/coe_xfr_sql_profile.sql Parameter1: sql_ID(required) Entervaluefor1:2ufquy7xs5nm5 PLAN_HASH_VALUEAVG_ET_SECS -------------------------- 1369807930.001 Parameter2: PLAN_HASH_VALUE(required) Entervaluefor2:1369807930 Valuespassedtocoe_xfr_sql_profile: ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ sql_ID :"2ufquy7xs5nm5" PLAN_HASH_VALUE:"1369807930" sql>BEGIN 2IF:sql_textISNULLTHEN 3 RAISE_APPLICATION_ERROR(-20100,'PLANforsql_ID&&sql_id.andPHV&&plan_hash_value.wasnotfoundinmemory(gv$sql_plan)orAWR(dba_hist_sql_plan).'); 4ENDIF; 5END; 6/ sql>SETTERMOFF; Executecoe_xfr_sql_profile_2ufquy7xs5nm5_1369807930.sql onTARGETsysteminordertocreateacustomsqlProfile withplan1369807930linkedtoadjustedsql_text. COE_XFR_sql_PROFILEcompleted.
从输出可以看出,生成一个名为coe_xfr_sql_profile_2ufquy7xs5nm5_1369807930.sql的脚本。
c. 把后生成的脚本里的Outline Data部分的Hint组合替换到先生成的脚本里,即下图红框部分内容
d. 执行coe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql脚本
zx@MYDB>@coe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql zx@MYDB>REM zx@MYDB>REM$Header:215187.1coe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql11.4.4.42017/02/28carlos.sierra$ zx@MYDB>REM zx@MYDB>REMCopyright(c)2000-2012,OracleCorporation.Allrightsreserved. zx@MYDB>REM zx@MYDB>REMAUTHOR zx@MYDB>REMcarlos.sierra@oracle.com zx@MYDB>REM zx@MYDB>REMSCRIPT zx@MYDB>REMcoe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql zx@MYDB>REM zx@MYDB>REMDESCRIPTION zx@MYDB>REMThisscriptisgeneratedbycoe_xfr_sql_profile.sql zx@MYDB>REMItcontainsthesql*Pluscommandstocreateacustom zx@MYDB>REMsqlProfileforsql_ID6chcc0pvvhqqmbasedonplanhash zx@MYDB>REMvalue3617692013. zx@MYDB>REMThecustomsqlProfiletobecreatedbythisscript zx@MYDB>REMwillaffectplansforsqlcommandswithsignature zx@MYDB>REMmatchingtheoneforsqlTextbelow. zx@MYDB>REMReviewsqlTextandadjustaccordingly. zx@MYDB>REM zx@MYDB>REMPARAMETERS zx@MYDB>REMNone. zx@MYDB>REM zx@MYDB>REMEXAMPLE zx@MYDB>REMsql>STARTcoe_xfr_sql_profile_6chcc0pvvhqqm_3617692013.sql; zx@MYDB>REM zx@MYDB>REMNOTES zx@MYDB>REM1.ShouldberunasSYSTEMorSYSDBA. zx@MYDB>REM2.UsermusthaveCREATEANYsqlPROFILEprivilege. zx@MYDB>REM3.SOURCEandTARGETsystemscanbethesameorsimilar. zx@MYDB>REM4.TodropthiscustomsqlProfileafterithasbeencreated: zx@MYDB>REMEXECDBMS_sqlTUNE.DROP_sql_PROFILE('coe_6chcc0pvvhqqm_3617692013'); zx@MYDB>REM5.BeawarethatusingDBMS_sqlTUNErequiresalicense zx@MYDB>REMfortheOracleTuningPack. zx@MYDB>REM6.IfyoumodifiedasqlputtingHintsinordertoproduceadesired zx@MYDB>REMPlan,youcanremovetheartificalHintsfromsqlTextpiecesbelow. zx@MYDB>REMBydoingsoyoucancreateacustomsqlProfilefortheoriginal zx@MYDB>REMsqlbutwiththePlancapturedfromthemodifiedsql(withHints). zx@MYDB>REM zx@MYDB>WHENEVERsqlERROREXITsql.sqlCODE; zx@MYDB>REM zx@MYDB>VARsignatureNUMBER; zx@MYDB>VARsignaturefNUMBER; zx@MYDB>REM zx@MYDB>DECLARE 2sql_txtCLOB; 3hSYS.sqlPROF_ATTR; 4PROCEDUREwa(p_lineINVARCHAR2)IS 5BEGIN 6DBMS_LOB.WRITEAPPEND(sql_txt,LENGTH(p_line),p_line); 7ENDwa; 8BEGIN 9DBMS_LOB.CREATETEMPORARY(sql_txt,TRUE); 10DBMS_LOB.OPEN(sql_txt,DBMS_LOB.LOB_READWRITE); 11--sqlTextpiecesbelowdonothavetobeofsamelength. 12--SoifyoueditsqlText(i.e.removingtemporaryHints),13--thereisnoneedtoeditorre-alignunmodifiedpieces. 14wa(q'[select/*+no_index(t1idx_t1)*/*fromt1wheren=1]'); 15DBMS_LOB.CLOSE(sql_txt); 16h:=SYS.sqlPROF_ATTR( 17q'[BEGIN_OUTLINE_DATA]',18q'[IGNORE_OPTIM_EMBEDDED_HINTS]',19q'[OPTIMIZER_FEATURES_ENABLE('11.2.0.1')]',20q'[DB_VERSION('11.2.0.1')]',21q'[ALL_ROWS]',22q'[OUTLINE_LEAF(@"SEL$1")]',23q'[INDEX(@"SEL$1""T1"@"SEL$1"("T1"."N"))]',24q'[END_OUTLINE_DATA]'); 25:signature:=DBMS_sqlTUNE.sqlTEXT_TO_SIGNATURE(sql_txt); 26:signaturef:=DBMS_sqlTUNE.sqlTEXT_TO_SIGNATURE(sql_txt,TRUE); 27DBMS_sqlTUNE.IMPORT_sql_PROFILE( 28sql_text=>sql_txt,29profile=>h,30name=>'coe_6chcc0pvvhqqm_3617692013',31description=>'coe6chcc0pvvhqqm3617692013'||:signature||''||:signaturef||'',32category=>'DEFAULT',33validate=>TRUE,34replace=>TRUE,35force_match=>FALSE/*TRUE:FORCE(matchevenwhendifferentliteralsinsql).FALSE:EXACT(similartoCURSOR_SHARING)*/); 36DBMS_LOB.FREETEMPORARY(sql_txt); 37END; 38/ PL/sqlproceduresuccessfullycompleted. zx@MYDB>WHENEVERsqlERRORCONTINUE zx@MYDB>SETECHOOFF; SIGNATURE --------------------- 3589138201450662673 SIGNATUREF --------------------- 8068435081012723673 ...manualcustomsqlProfilehasbeencreated COE_XFR_sql_PROFILE_6chcc0pvvhqqm_3617692013completed
e. 执行完成后再次查看目标sql的执行计划
zx@MYDB>select/*+no_index(t1idx_t1)*/*fromt1wheren=1; N ---------- 1
从执行计划中可以看出已经走了INDEX RANGE SCAN,而且note部分提示sql profile coe_6chcc0pvvhqqm_3617692013 used for this statement,说明执行sql时使用了该sql Profile。
如果想在目标sql的sql文本发生变动时sql Profile依然生效,则需要修改生成的脚本里的force_match=>true。
参考:《基于Oracle的sql优化》
官方文档:http://docs.oracle.com/cd/E11882_01/server.112/e41573/sql_tune.htm#PFGRF94854
http://docs.oracle.com/cd/E11882_01/appdev.112/e40758/d_sqltun.htm#CHDGAJCI