Oracle中的Hint可以用來調整SQL的執行計劃,提高SQL執行效率。下面分類介紹Oracle數據庫中常見的Hint。這裡主要介紹Oracle11gR2中的常見Hint。像我們在做優化的時候,如果一個sql執行效率比較低,我們可以用不同的hint來看下執行計劃,對比選擇最優的hint,如果CBO選擇了不好的執行計劃,就要考慮是不是沒對表做統計分析了...
1、ALL_ROWS
ALL_ROWS是針對整個目标SQL的Hint,它的含義是讓優化器啟用CBO,而且在得到目标SQL的執行計劃時會選擇那些吞吐量最佳的執行路徑。這裡的“吞吐量最佳”是指資源消耗量(即對I/O、CPU等硬件資源的消耗量)最小,也就是說在ALL_ROWS Hint生效的情況下,優化器會啟用CBO而且會依據各個執行路徑的資源消耗量來計算它們各自的成本。
ALL_ROWS Hint的格式如下:
/* ALL_ROWS */
set line 200 set autotrace on select /* all_rows */ empno,ename,sal,job from emp where empno=7396;
從Oracle10g開始,ALL_ROWS就是默認的優化器模式,啟用的就是CBO。
show parameter optimizer_mode;
如果目标SQL中除了ALL_ROWS之外還使用了其他與執行路徑、表連接相關的Hint,優化器會優先考慮ALL_ROWS。
2、FIRST_ROWS(n)
FIRST_ROWS(n)是針對整個目标SQL的Hint,它的含義是讓優化器啟用CBO模式,而且在得到目标SQL的執行計劃時會選擇那些能以最快的響應時間返回頭n條記錄的執行路徑,也就是說在FIRST_ROWS(n) Hint生效的情況下,優化器會啟用CBO,而且會依據返回頭n條記錄的響應時間來決定目标SQL的執行計劃。
FIRST_ROWS(n)格式如下:
/* FIRST_ROWS(n) */
select /* first_rows(10) */ empno,ename,sal,job from emp where deptno=30;
上述SQL中使用了/* first_rows(10) */,其含義是告訴優化器我們想以最短的響應時間返回滿足條件"deptno=30"的前10條記錄,并不是隻返回10條,這裡需要注意。
注意,FIRST_ROWS(n) Hint和優化器模式FIRST_ROWS_n不是一一對應的。優化器模式FIRST_ROWS_n中的n隻能是1、10、100、1000。但FIRST_ROWS(n) Hint中的n還可以是其他值。
3、RULE(基本不用)
RULE是針對整個目标SQL的Hint,它表示對目标SQL啟用RBO。
格式如下:
/* RULE */
select /* rule */ empno,ename,sal,job from emp where deptno=30;
RULE不能與除DRIVING_SITE以外的Hint聯用,當RULE與除DRIVING_SITE以外的Hint聯用時,其他Hint可能會失效;當RULE與DRIVING_SITE聯用時,它自身可能會失效,所以RULE Hint最好是單獨使用。
一般情況下,并不推薦使用RULE Hint。一來是因為Oracle早就不支持RBO了,二來啟用RBO後優化器在執行目标SQL時可選擇的執行路徑将大大減少,很多執行路徑RBO根本就不支持(比如哈希連接),就也就意味着啟用RBO後目标SQL跑出正确執行計劃的概率将大大降低。
1、FULL
FULL是針對單個目标表的Hint,它的含義是讓優化器對目标表執行全表掃描。
格式如下:
/* FULL(目标表) */
select /* full(emp) */ empno,ename,sal,job from emp where deptno=30;
上述SQL中Hint的含義是讓優化器對目标表EMP執行全表掃描操作,而不考慮走表EMP上的任何索引(即使列EMPNO上有主鍵索引)。
2、ROIWD
ROIWD是針對單個目标表的Hint,它的含義是讓優化器對目标表執行RWOID掃描。隻有目标SQL中使用了含ROWID的where條件時ROWID Hint才有意義。
格式如下:
/* ROWID(目标表) */
select /* rowid(emp) */ empno,ename,sal,job from emp where rowid='AAAR3xAAEAAAACXAAN';
Oracle 11gR2中即使使用了ROWID Hint,Oracle還是會将讀到的塊緩存在Buffer Cache中。
1、INDEX
INDEX是針對單個目标表的Hint,它的含義是讓優化器對目标表的的目标索引執行索引掃描操作。
INDEX Hint中的目标索引幾乎可以是Oracle數據庫中所有類型的索引(包括B樹索引、位圖索引、函數索引等)。
INDEX Hint的模式有四種:
格式1 /* INDEX(目标表 目标索引) */ 格式2 /* INDEX(目标表 目标索引1 目标索引2 …… 目标索引n) */ 格式3 /* INDEX(目标表 (目标索引1的索引列名) (目标索引2的索引列名) …… (目标索引n的索引列名)) */ 格式4 /* INDEX(目标表) */
格式1表示僅指定了目标表上的一個目标索引,此時優化器隻會考慮對這個目标索引執行索引掃描操作,而不會去考慮全表掃描或者對該目标表上的其他索引執行索引掃描操作。
格式2表示指定了目标表上的n個目标索引,此時優化器隻會考慮對這n個目标索引執行索引掃描操作,而不會去考慮全表掃描或者對該目标表上的其他索引執行索引掃描操作。注意,優化器在考慮這n個目标索引時,可能是分别計算出單獨掃描各個目标索引的成本後,再選擇其中成本值最低的索引;也可能是先分别掃描目标索引中的兩個或多個索引,然後再對掃描結果執行合并操作。當然,後面這種可能性的前提條件是優化器計算出來這樣做的成本值是最低的。
格式三也是表是指定了目标表上的n個目标索引,隻不過此時是用指定目标索引的索引列名來代替對應的目标索引名。如果目标索引是複合索引,則在用于指定該索引列名的括号内也可以指定該目标索引的多個索引列,各個索引列之間用空格分隔就可以了。
格式的表示指定了目标表上所有已存在的索引,此時優化器隻會考慮對該目标表上所有已存在的索引執行索引掃描操作,而不會去考慮全表掃描操作。注意,這裡優化器在考慮該目标表上所有已存在的索引時,可能是分别計算出單獨掃描這些索引的成本後再選擇其中成本值最低的索引;也可能是先分别掃描這些索引中的兩個或多個索引,然後再對掃描結果執行合并操作。當然,後面這種可能性的前提條件是優化器計算出來這樣做的成本值是最低的。
select /* index(emp pk_emp) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* index(emp pk_emp idx_emp_mgr idx_emp_dept) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* index(emp (empno) (mgr) (deptno)) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* index */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20;
2、NO_INDEX
NO_INDEX是針對單個目标表的Hint,它是INDEX的反義Hint,其含義是讓優化器不對目标表上的目标索引執行掃描操作。
INDEX Hint中的目标索引也幾乎可以是Oracle數據庫中所有類型的索引(包括B樹索引、位圖索引、函數索引等)。
格式有如下三種:
格式1 /* NO_INDEX(目标表 目标索引) */ 格式2 /* NO_INDEX(目标表 目标索引1 目标索引2 …… 目标索引n) */ 格式3 /* NO_INDEX(目标表) */
格式1表示僅指定了目标表上的一個目标索引,此時優化器隻是不會考慮對這個目标索引執行索引掃描操作,但還是會考慮全表掃描或者對該目标表上的其他索引執行索引掃描操作。
格式2表示指定了目标表上的n個目标索引,此時優化器隻是不會考慮對這n個目标索引執行索引掃描操作,但還是會考慮全表掃描或者對該目标表上的其他索引執行索引掃描操作。
格式3表示指定了目标表上的所有已存在的索引,即此時優化器不會考慮對該目标表上所有已存在的索引執行索引掃描操作,這相當于對目标表指定了全表掃描。
select /* no_index(emp pk_emp) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* no_index(emp pk_emp idx_emp_mgr idx_emp_dept) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* no_index */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20;
2、INDEX_DESC
INDEX_DESC是針對單個目标表的Hint,它的含義是讓優化器對目标表上的目标索引執行索引降序掃描操作。如果目标索引是升序的,則INDEX_DESC Hint會使Oracle以降序的方式掃描該索引;如果目标索引是降序的,則INDEX_DESC Hint會使Oracle以升序的方式掃描該索引。
格式有三種:
格式1 /* INDEX_DESC(目标表 目标索引) */
格式2 /* INDEX_DESC(目标表 目标索引1 目标索引2 …… 目标索引n) */
格式3 /* INDEX_DESC(目标表) */
select /* index_desc(emp pk_emp) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* index_desc(emp pk_emp idx_emp_mgr idx_emp_dept) */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20; select /* index_desc */ empno,ename,sal,job from emp where empno=7369 and mgr=7902 and deptno=20;
3、INDEX_JOIN
INDEX_JOIN是針對單個目标表的Hint,它的含義是讓優化器對目标表上的多個目标索引執行INDEX JOIN操作。INDEX JOIN能成立的前提條件是SELECT語句中所有的查詢列都存在于目标表上的多個目标索引中,即通過掃描這些索引就可以得到所有的查詢列而不用回表。
格式如下:
格式1 /* INDEX_JOIN(目标表 目标索引1 目标索引2 …… 目标索引n) */ 格式2 /* INDEX_JOIN */
上述兩種格式的含義與INDEX_COMBINE Hint中對應格式的含義相同。
select /* index_join(emp pk_emp idx_emp_mgr) */ empno,mgr from emp where empno>7369 and mgr<7902; select /* index_join(emp) */ empno,mgr from emp where empno>7369 and mgr<7902;
實例:
select empno,mgr from emp where empno>7369 and mgr<7902; select /* index_join(emp) */ empno,mgr from emp where empno>7369 and mgr<7902;
1、ORDERED
ORDERED是針對多個目标表的Hint,它的含義是讓優化器對多個目标表執行表連接操作時,執照它們在目标SQL的where條件中出現的順序從左到右依次進行連接。
格式如下:
/* ORDERED */
select /* ordered */ e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' order by e.ename;
實例:
select e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' order by e.ename; select /* ordered */ e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' order by e.ename;
從上面的執行計劃可以看出不使用ordered Hint時表掃描的順序是DEPT->EMP->JOBS,但是使用ordered Hint後,表掃描的順序變為了EMP->JOBS->DEPT與目标SQL中的順序一緻了,在修改了目标SQL文本之後表的掃描順序也相應地變為了EMP->DEPT->JOBS。
2、LEADING
LEADING是針對多個目标表的Hint,它的含義是讓優化器将我們指定的多個表的連接結果作為目标SQL表連接過程中的驅動結果集,并且将LEADING Hint中從左至右出現的第一個目标表作為整個表連接過程中的首個驅動表。
LEADING比ORDERED要溫和一些,因為它隻是指定了首個驅動表和驅動結果集,沒有像ORDERED那樣完全指定了表連接的順序,也就是說LEADING給了優化器更大的調整餘地。
當LEADING Hint中指定的表并不能作為目标SQL的連接過程中的驅動表或者驅動結果集時,Oracle會忽略該Hint。
格式如下:
/* LEADING(目标表1 目标表2 …… 目标表n) */
select /* leading(t e) */ e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d,emp_temp t where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' and e.ename=t.ename order by e.ename;
實例:
select e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d,emp_temp t where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' and e.ename=t.ename order by e.ename; select /* leading(t e) */ e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d,emp_temp t where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' and e.ename=t.ename order by e.ename; select /* ordered */ e.ename,j.job,e.sal,d.deptno from emp e,jobs j,dept d,emp_temp t where e.empno=j.empno and e.deptno=d.deptno and d.loc='CHICAGO' and e.ename=t.ename order by e.ename;
從上面的執行計劃可以看出不使用Hint時表掃描順序是DEPT->EMP->JOBS->EMP_TEMP;使用LEADING Hint時表掃描順序是EMP_TEMP->EMP->DEPT->JOBS,EMP_TEMP做首個驅動表和表EMP的連接結果做為驅動結果集,與Hint要求一緻。;使用Ordered Hint時表掃描順序是EMP->JOBS->DEPT->EMP_TEMP,與SQL中順序一緻。
篇幅有限,關于oracle常見的一些hint就介紹到這了,如果大家想深入學習數據庫的話hint這一塊内容還是必不可少的,特别是做優化的時候。
後面會分享更多關于DBA方面内容,感興趣的朋友可以關注下!
,
更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!