oracle 索引介绍、作用、使用

oracle 索引介绍、作用、使用


1、什么是索引?

  1. 索引是建立在表的一列或者多列上的辅助对象,目的是加快访问表中的数据。

  2. oracle 存储索引的结构是B*数(平衡树),而索引是由根节点、分支点和叶子点组成,上级索引块包含下级索引块的索引数据,叶节点包含索引数据和确定行实际位置的rowid。

2、索引说明

  • 索引是数据库对象之一,用于加快数据的检索,类似于书籍的索引。在数据库中索引可以减少数据库程序查询结果时需要读取的数据量,类似于在书籍中我们利用索引可以不用翻阅整本书即可找到想要的信息。

  • 索引是建立在表上的可选对象;索引的关键在于通过一组排序后的索引键来取代默认的全表扫描检索方式,从而提高检索效率。

  • 索引在逻辑上和物理上都与相关的表和数据无关,当创建或者删除一个索引时,不会影响基本的表。

  • 索引一旦建立,在表上进行DML操作时(例如在执行插入、修改或者删除相关操作时),oracle会自动管理索引,索引删除,不会对表产生影响。

  • 索引对用户是透明的,无论表上是否有索引,sql语句的用法不变。

  • oracle创建主键时会自动在该列上创建索引。

3、索引原理

  1. 若没有索引,搜索某个记录时(例如查找name=’wish’)需要搜索所有的记录,因为不能保证只有一个wish,必须全部搜索一遍
  2. 若在name上建立索引,oracle会对全表进行一次搜索,将每条记录的name值哪找升序排列,然后构建索引条目(name和rowid),存储到索引段中,查询name为wish时即可直接查找对应地方
    3.创建了索引并不一定就会使用,oracle自动统计表的信息后,决定是否使用索引,表中数据很少时使用全表扫描速度已经很快,没有必要使用索引。

4、索引创建

CREATE [UNIQUE] | [BITMAP] INDEX index_name  --unique表示唯一索引
ON table_name([column1 [ASC|DESC],column2    --bitmap,创建位图索引
[ASC|DESC],…] | [express])
[TABLESPACE tablespace_name]
[PCTFREE n1]                                 --指定索引在数据块中空闲空间
[STORAGE (INITIAL n2)]
[NOLOGGING]                                  --表示创建和重建索引时允许对表做DML操作,默认情况下不应该使用
[NOLINE]
[NOSORT];                                    --表示创建索引时不进行排序,默认不适用,如果数据已经是按照该索引顺序排列的可以使用

例子 : create index index_name on table_name([column1 [ASC|DESC])

5、修改索引

1、重命名索引

alter index index_sno rename to bitmap_index;

2、合并索引(表使用一段时间后在索引中会产生碎片,此时索引效率会降低,可以选择重建索引或者合并索引,合并索引方式更好些,无需额外存储空间,代价较低)

alter index index_sno coalesce;

3、重建索引

  方式一:删除原来的索引,重新建立索引

  方式二:alter index index_sno rebuild;
  

6、删除索引

drop index index_sno;

7、查看索引

  SELECT * FROM ALL_INDEXES WHERE TABLE_NAME='表名';

7、索引分类

1. B树索引(默认索引,保存讲过排序过的索引列和对应的rowid值)

1)说明:

 1. oracle中最常用的索引;B树索引就是一颗二叉树;叶子节点(双向链表)包含索引列和指向表中每个匹配行的ROWID值.

 2. 所有叶子节点具有相同的深度,因而不管查询条件怎样,查询速度基本相同.

 3. 能够适应精确查询、模糊查询和比较查询.

2)分类:

UNIQUE,NON-UNIQUE(默认),REVERSE KEY(数据列中的数据是反向存储的)

3)创建例子

craete index index_sno on student('sno');

4)适合使用场景:

列基数(列不重复值的个数)大时适合使用B数索引

2. 位图索引

1)说明:

1.创建位图索引时,oracle会扫描整张表,并为索引列的每个取值建立一个位图(位图中,对表中每一行 使用一位(bit,0或者1)来标识该行是否包含该位图的索引列的取值,如果为1,表示对应的rowid所在的记录包含该位图索引列值),最后通过位图索 引中的映射函数完成位到行的ROWID的转换

2)创建例子

create bitmap index index_sno on student(sno);

3) 适合场景:

对于基数小的列适合简历位图索引(例如性别等)

3.单列索引和复合索引(基于多个列创建)

1) 注意:

即如果索引建立在多个列上,只有它的第一个列被where子句引用时,优化器才会使用该索引,即至少要包含组合索引的第一列
  1. 函数索引

1)说明:

1. 当经常要访问一些函数或者表达式时,可以将其存储在索引中,这样下次访问时,该值已经计算出来了,可以加快查询速度

2. 函数索引既可以使用B数索引,也可以使用位图索引;当函数结果不确定时采用B树索引,结果是固定的某几个值时使用位图索引

3. 函数索引中可以水泥用len、trim、substr、upper(每行返回独立结果),不能使用如sum、max、min、avg等

2)例子:

create index fbi  on student (upper(name));
select * from student where upper(name) ='WISH';

8、索引建立原则总结

1. 如果有两个或者以上的索引,其中有一个唯一性索引,而其他是非唯一,这种情况下oracle将使用唯一性索引而完全忽略非唯一性索引。

2. 至少要包含组合索引的第一列(即如果索引建立在多个列上,只有它的第一个列被where子句引用时,优化器才会使用该索引)。

3. 小表不要简历索引。

 4. 对于基数大的列适合建立B树索引,对于基数小的列适合简历位图索引。

5. 列中有很多空值,但经常查询该列上非空记录时应该建立索引。

6. 经常进行连接查询的列应该创建索引。

7. 使用create index时要将最常查询的列放在最前面。

8. LONG(可变长字符串数据,最长2G)和LONG RAW(可变长二进制数据,最长2G)列不能创建索引。

9.限制表中索引的数量(创建索引耗费时间,并且随数据量的增大而增大;索引会占用物理空间;当对表中的数据进行增加、删除和修改的时候,索引也要动态的维护,降低了数据的维护速度)。

注意事项

  1. 通配符在搜索词首出现时,oracle不能使用索引,eg:

–我们在name上创建索引;

 create index index_name on student('name');

–下面的方式oracle不适用name索引

select * from student where name like '%wish%';

–如果通配符出现在字符串的其他位置时,优化器能够利用索引;如下:

select * from student where name like 'wish%';
  1. 不要在索引列上使用not,可以采用其他方式代替如下:(oracle碰到not会停止使用索引,而采用全表扫描)。

select * from student where not (score=100);

select * from student where score <> 100;

–替换为

select * from student where score>100 or score <100
  1. 索引上使用空值比较将停止使用索引, eg:

    select * from student where score is not null;

常见的索引限制问题:

1、使用不等于操作符(<>, !=)

下面这种情况,即使在列dept_id有一个索引,查询语句仍然执行一次全表扫描

select * from dept where staff_num <> 1000;

但是开发中的确需要这样的查询,难道没有解决问题的办法了吗?

有!

通过把用 or 语法替代不等号进行查询,就可以使用索引,以避免全表扫描:上面的语句改成下面这样的,就可以使用索引了。

select * from dept shere staff_num < 1000 or dept_id > 1000;

2、使用 is null 或 is not null

使用 is null 或is nuo null也会限制索引的使用,因为数据库并没有定义null值。如果被索引的列中有很多nu3,就不会使用这个索引(除非索引是一个位图索引,关于位图索引,会在以后的blog文章里做详细解释)。在sql语句中使用null会造成很多麻烦。

解决这个问题的办法就是:建表时把需要索引的列定义为非空(not null)

4、使用函数

如果没有使用基于函数的索引,那么where子句中对存在索引的列使用函数时,会使优化器忽略掉这些索引。下面的查询就不会使用索引:

select * from staff where trunc(birthdate) = '01-MAY-82';

但是把函数应用在条件上,索引是可以生效的,把上面的语句改成下面的语句,就可以通过索引进行查找。

select * from staff where birthdate < (to_date('01-MAY-82') + 0.9999);

5、比较不匹配的数据类型

比较不匹配的数据类型也是难于发现的性能问题之一。下面的例子中,dept_id是一个varchar2型的字段,在这个字段上有索引,但是下面的语句会执行全表扫描。

select * from dept where dept_id = 900198;

这是因为oracle会自动把where子句转换成to_number(dept_id)=900198,就是3所说的情况,这样就限制了索引的使用。把SQL语句改为如下形式就可以使用索引

select * from dept where dept_id = '900198';

6、使用like子句

使用like子句查询时,数据需要把所有的记录都遍历来进行判断,索引不能发挥作用,这种情况也要尽量避免。

Like 的字符串中第一个字符如果是‘%’则用不到索引

Column1 like ‘aaa%’ 是可以的
Column1 like ‘%aaa%’用不到

7.使用IN

尽管In写法要比exists简单一些,exists一般来说性能要比In要高的多
用In还是用Exists的时机
当in的集合比较小的时候,或者用Exists无法用到选择性高的索引的时候,用In要好,否则就要用Exists

例:select count(*) from person_info where xb in (select xb_id from dic_sex);
Select count(*) from n_acntbasic a where shbxdjm =:a and exists(select 1 from person_info where pid=a.pid and …);

Select * from person_info where zjhm=3101….;将会对person_info全表扫描
Select * from person_info where zjhm =‘3101…’才能用到索引

假定TEST表的dt字段是date类型的并且对dt建了索引。
如果要查‘20041010’一天的数据.下面的方法用不到索引

Select * from test where to_char(dt,’yyyymmdd’) =‘20041010’;
而select * from test where dt >=to_date(‘20041010’,’yyyymmdd’) and dt < to_date(‘20041010’,’yyyymmdd’) + 1 将会用到索引。

8.如果能不用到排序,则尽量避免排序。
用到排序的情况有

集合操作。Union ,minus ,intersect等,注:union all 是不排序的。
Order by
Group by
Distinct

In 有时候也会用到排序
确实要排序的时候也尽量要排序小数据量
,尽量让排序在内存中执行,有文章说,内存排序的速度是硬盘排序的1万倍。 在排序的字段上创建索引,让排序在内存中执行,加快排序速度。

16.在基于CBO的优化器(花费)下,表的统计数据过期。也可能导致不使用索引。

   解决:执行表分析。获取表的最新信息。

17.获取的数据量过大,全部扫描效率更高

18.索引字段的值分散率太低,值太集中,如类型字段都是1,2, 状态类型Y-有效/N-无效。这类型的字段最好别建索引。尽管在这些字段上建立了索引,但对全表数据区分度不大。最后还是会全表扫描。

作者 [@GHQ][3]
2018 年 06月 06日

参考文章链接

  • 11
    点赞
  • 76
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Oracle索引是一种数据结构,它可以加快查询的速度。使用索引可以优化查询,使其更快,同时可以减少查询所需的资源。以下是使用Oracle索引的步骤: 1. 创建索引 可以使用CREATE INDEX语句创建一个索引,例如: ``` CREATE INDEX idx_name ON table_name(column_name); ``` 其中,idx_name是索引的名称,table_name是要创建索引的表名,column_name是要创建索引的列名。 2. 查询优化 当执行SELECT语句时,Oracle会自动选择最佳的执行计划。如果有索引可以使用Oracle使用索引来优化查询。例如: ``` SELECT * FROM table_name WHERE column_name = 'value'; ``` 如果存在一个索引idx_name,包含了column_name列,那么Oracle使用索引来优化查询。 3. 索引维护 当表中的数据发生变化时,索引也需要进行相应的维护。例如,当插入一行数据时,Oracle会自动更新索引。如果删除一行数据,则索引也需要更新。 4. 索引类型 Oracle提供了多种类型的索引,包括B-tree、位图和哈希等。不同类型的索引适用于不同的场景,可以根据具体情况选择合适的索引类型。 5. 索引管理 可以使用Oracle提供的索引管理工具,例如Oracle Enterprise Manager或SQL Developer等,来管理索引。可以查看索引的状态、性能以及使用情况等信息,并进行索引优化和维护工作。 总之,使用Oracle索引可以加快查询速度,提高数据库性能。但是,在创建索引时需要谨慎,因为过多的索引会增加数据库的维护成本,并且可能会降低查询性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值