欢迎投稿

今日深度:

Oracle优化之Oracle访问数据的几种方式详解,orac

Oracle优化之Oracle访问数据的几种方式详解,oracle访问数据


ORACLE如何访问数据

1.全表扫描(FullTableScansFTS)

为实现全表扫描, Oracle读取表中所有行,并检查每一行是否满足语句的WHERE限制条件一个多块读操作可以使一次I/O能读取多块数据块(db_block_multiblock_read_count参数设定), 而不是只读取一个数据块, 这极大的减 少了I/O总次数, 提高了系统的吞吐量,所以利用多块读的方法可以十分高效地实现全表扫描,而且只有在全表扫描的情况下才能使用多块读操作。在这种访问模式下,每个数据块只被读一次。使用FTS的前提条件:在较大的表上不建议使用全表扫描,除非取出数据的比较多,超过总量的5%——10%,或你

想使用并行查询功能时。

Example:

SQL>explain plan for select * fromdual;

QueryPlan

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

SELECTSTATEMENT[CHOOSE]Cost=

TABLEACCESSFULLDUAL

全表扫描模式下会读数据到表的高水位线( HWM即表示表曾经扩展的最后一个数据

块),读取速度依赖于Oracle初始化参数db_block_multiblock_read_count

(FTS扫描会使表使用上升到高水位(HWM),HWM标识了表最后写入数据的块,如果你

用DELETE删除了所有的数据表仍然处于高水位(HWM),只有用 TRUNCATE才能使表回归,FTS使用多IO从磁盘读取数据块).

QueryPlan

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

SELECTSTATEMENT[CHOOSE]Cost=1

**INDEXUNIQUESCANEMP_I1 --如果索引里就找到了所要的数据,就不会再去访问表

通过设置db_block_multiblock_read_count可以设置一次IO能读取的数据块个数, 从而有效减少全表扫描时的IO总次数, 也就是通过预读机制将将要访问的数据块预先读入内存中。只有在全表扫描情况下才能使用多块读操作。

2.通过ROWID的表存取(TableAccessbyROWID或rowidlookup)

行的ROWID指出了该行所在的数据文件、数据块以及行在该块中的位置,所以通过

ROWID来存取数据可以快速定位到目标数据上,是 Oracle存取单行数据的最快方法。

这种存取方法不会用到多块读操作, 一次I/O只能读取一个数据块。 我们会经常在执行计划中看到该存取方法,如通过索引查询数据。

使用ROWID存取的方法:

SQL>explainplanforselect*fromdeptwhererowid=''AAAAyGAADAAAAATAAF'';

QueryPlan

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

SELECTSTATEMENT[CHOOSE]Cost=1

TABLEACCESSBYROWIDDEPT[ANALYZED]

3. 索引扫描(IndexScan或indexlookup)

我们先通过index查找到数据对应的rowid值 ( 对于非唯一索引可能返回多个rowid值),然后根据rowid直接从表中得到具体的数据,这种查找方式称为索引扫描或索引查找 ( indexlookup)。 一个 rowid唯一的表示一行数据, 该行对应的数据块是通过一次 I/0得到的, 在此情况下该次 i/o只会读取一个数据库块。

在索引中,除了存储每个索引的值外,索引还存储具有此值的行对应的 ROWID值。

索引扫描可以由 2步组成:

1 扫描索引得到对应的 rowid值。

2 通过找到的 rowid从表中读出具体的数据。

每步都是单独的一次 I/O, 但是对于索引, 由于经常使用, 绝大多数都已经 CACHE到内存中, 所以第 1步的 I/O经常是逻辑 I/O, 即数据可以从内存中得到。 但是对于第 2步来说, 如果表比较大, 则其数据不可能全在内存中, 所以其 I/O很有可能是物理 I/O, 这 是一个机械操作, 相对逻辑

I/O

是极其费时间的。

所以如果多大表进行索引扫描, 取出的数据如果大于总量的 5% —— 10%

,使用索引

扫描会效率下降很多。如下列所示:

SQL> explain plan for select empno, ename from emp where empno=10;

Query Plan

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

SELECT STATEMENT [CHOOSE] Cost=1

TABLE ACCESS BY ROWID EMP [ANALYZED]

INDEX UNIQUE SCAN EMP_I1

但是如果查询的数据能全在索引中找到,就可以避免进行第 2步操作,避免了不必要

的 I/O,此时即使通过索引扫描取出的数据比较多,效率还是很高的

SQL> explain plan for select empno from emp where empno=10;-- 只查询 empno列

Query Plan

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

SELECT STATEMENT [CHOOSE] Cost=1

INDEX UNIQUE SCAN EMP_I1

进一步讲, 如果 sql语句中对索引列进行排序, 因为索引已经预先排序好了, 所以在执行计划中不需要再对索引列进行排序

SQL> explain plan for select empno, ename from emp where empno > 7876

order by empno;

Query Plan

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

SELECT STATEMENT[CHOOSE] Cost=1

TABLE ACCESS BY ROWID EMP [ANALYZED]

INDEX RANGE SCAN EMP_I1 [ANALYZED]

从这个例子中可以看到: 因为索引是已经排序了的, 所以将按照索引的顺序查询出符合条件的行,因此避免了进一步排序操作。

根据索引的类型与 where限制条件的不同,有 4种类型的索引扫描:

1) 索引唯一扫描(index unique scan)

通过唯一索引查找一个数值经常返回单个ROWID.如果存在 UNIQUE 或 PRIMARY

KEY 约束(它保证了语句只存取单行)的话, Oracle经常实现唯一性扫描。

使用唯一性约束的例子:

SQL> explain plan for select empno, ename from emp where empno=10;

Query Plan

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

SELECTSTATEMENT[CHOOSE] Cost=1

TABLEACCESS BYROWID EMP [ANALYZED]

INDEX UNIQUE SCAN EMP_I1

2) 索引范围扫描(index range scan)

使用一个索引存取多行数据,在唯一索引上使用索引范围扫描的典型情况下是在谓词

( where限制条件)中使用了范围操作符(如>、 <、 <>、 >=、 <=、 between)

使用索引范围扫描的例子:

SQL> explain plan for select empno, ename from emp where empno > 7876

orderby empno;

Query Plan

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

SELECTSTATEMENT[CHOOSE] Cost=1

TABLEACCESS BYROWID EMP [ANALYZED]

INDEX RANGE SCAN EMP_I1 [ANALYZED]

在非唯一索引上,谓词col = 5可能返回多行数据,所以在非唯一索引上都使用索引范

围扫描。

使用index rang scan的3种情况:

1 在唯一索引列上使用了range操作符( > < <> >= <= between)

2 在组合索引上,只使用部分列进行查询,导致查询出多行

3 对非唯一索引列上进行的任何查询。

3) 索引全扫描(index full scan)

与全表扫描对应, 也有相应的全索引扫描。而且此时查询出的数据都必须从索引中可以直接得到。

全索引扫描的例子:

SQL>explain plan for select empno, ename from big_emp order by empno,ename;

Query Plan

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

SELECTSTATEMENT[CHOOSE] Cost=26

INDEX FULLSCAN BE_IX [ANALYZED]

4) 索引快速扫描(index fastfull scan)

扫描索引中的所有的数据块,在这种存取方法中, 可以使用多块读功能,也可以使用并行读入,以便获得最大吞吐量与缩短执行时间。

与 index full scan很类似,但是一个显著的区别就是它不对查询出的数据进行排序, 即数据不是以排序顺序被返回。

索引快速扫描的例子: BE_IX索引是一个多列索引: big_emp( empno, ename)

SQL> explain plan for select empno, ename from big_emp;

Query Plan

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

SELECT STATEMENT[CHOOSE] Cost=1

INDEX FAST FULLSCAN BE_IX [ANALYZED]

只选择多列索引的第2列:

SQL> explain plan for select ename from big_emp;

Query Plan

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

SELECT STATEMENT[CHOOSE] Cost=1

INDEX FAST FULLSCAN BE_IX [ANALYZED]

www.htsjk.Com true http://www.htsjk.com/oracle/24374.html NewsArticle Oracle优化之Oracle访问数据的几种方式详解,oracle访问数据 ORACLE如何访问数据 1.全表扫描(FullTableScansFTS) 为实现全表扫描, Oracle读取表中所有行,并检查每一行是否满足语句的WHERE限制条...
评论暂时关闭