高手闲谈Oracle索引扫描

数据库 Oracle
这里介绍即使通过Oracle索引扫描取出的数据比较多,效率还是很高的,因为这只会在索引中读取。

Oracle还是比较常用的,于是我研究了一下Oracle索引扫描,在这里拿出来和大家分享一下,希望对大家有用。我们先通过index查找到数据对应的rowid值(对于非***索引可能返回多个rowid值),然后根据rowid直接从表中得到具体的数据,这种查找方式称为Oracle索引扫描或索引查找(index lookup)。一个rowid***的表示一行数据,该行对应的数据块是通过一次i/o得到的,在此情况下该次i/o只会读取一个数据库块。

在索引中,除了存储每个索引的值外,索引还存储具有此值的行对应的ROWID值。Oracle索引扫描可以由2步组成:
(1) 扫描索引得到对应的rowid值。
(2) 通过找到的rowid从表中读出具体的数据。

每步都是单独的一次I/O,但是对于索引,由于经常使用,绝大多数都已经CACHE到内存中,所以第1步的I /O经常是逻辑I/O,即数据可以从内存中得到。但是对于第2步来说,如果表比较大,则其数据不可能全在内存中,所以其I/O很有可能是物理I/O,这是一个机械操作,相对逻辑I/O来说,是极其费时间的。所以如果多大表进行索引扫描,取出的数据如果大于总量的5% -- 10%,使用索引扫描会效率下降很多。

如下列所示:

  1. SQL> explain plan for select empno, ename from emp where empno=10;  
  2. Query Plan  
  3. SELECT STATEMENT [CHOOSE] Cost=1 
  4. TABLE ACCESS BY ROWID EMP [ANALYZED]  
  5. INDEX UNIQUE SCAN EMP_I1 

注意TABLE ACCESS BY ROWID EMP部分,这表明这不是通过FTS存取路径访问数据,而是通过rowid lookup存取路径访问数据的。在此例中,所需要的rowid是由于在索引查找empno列的值得到的,这种方式是INDEX UNIQUE SCAN查找,后面给予介绍,EMP_I1为使用的进行索引查找的索引名字。

但是如果查询的数据能全在索引中找到,就可以避免进行第2步操作,避免了不必要的I/O,此时即使通过Oracle索引扫描取出的数据比较多,效率还是很高的,因为这只会在索引中读取。所以上面我在介绍基于规则的优化器时,使用了select count(id) from SWD_BILLDETAIL where cn <'6',而没有使用select count(cn) from SWD_BILLDETAIL where cn <'6'。因为在实际情况中,只查询被索引列的值的情况极为少,所以,如果我在查询中使用count(cn),则不具有代表性。

  1. SQL> explain plan for select empno from emp where empno=10; -- 只查询empno列值  
  2. Query Plan  
  3. SELECT STATEMENT [CHOOSE] Cost=1 
  4. INDEX UNIQUE SCAN EMP_I1 

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

  1. SQL> explain plan for select empno, ename from emp  
  2. where empno > 7876 order by empno;  
  3. Query Plan  
  4. SELECT STATEMENT [CHOOSE] Cost=1 
  5. TABLE ACCESS BY ROWID EMP [ANALYZED]  
  6. INDEX RANGE SCAN EMP_I1 [ANALYZED] 

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

【编辑推荐】

  1. 代码讲解Oracle可选择性
  2. Oracle管理工具Oracle SQL Handler功能一览
  3. 美国司法部公开支持Oracle收购Sun
  4. 闲谈Oracle执行计划的步骤顺序
  5. 六分钟学会Oracle全表扫描
责任编辑:佚名 来源: cnbeta
相关推荐

2009-11-16 17:33:21

重做Oracle日志文

2009-04-10 14:38:17

Oracle高手位图索引

2010-04-08 10:57:04

Oracle编程

2010-11-15 15:20:13

Oracle索引扫描

2009-11-11 09:40:53

Oracle索引扫描

2010-04-07 10:02:31

2009-11-10 16:00:05

Oracle执行计划

2011-07-19 13:08:10

2009-11-17 10:20:01

Oracle Spat

2009-11-17 15:59:25

Oracle物化视图

2010-04-14 16:22:42

Oracle 9i

2009-10-29 16:57:05

Oracle传输表空间

2010-04-14 16:45:29

Oracle 9i全索

2010-04-15 16:16:57

Oracle数据库应用

2024-08-29 08:11:56

PostgresOracleMySQL

2009-11-02 18:56:49

调整Oracle数据库

2009-11-17 13:49:46

Oracle prim

2010-10-26 17:34:03

Oracle索引

2011-03-11 16:32:33

2010-03-17 14:26:40

云计算
点赞
收藏

51CTO技术栈公众号