• 设为首页
  • 收藏本站
  • 积分充值
  • VIP赞助
  • 手机版
  • 微博
  • 微信
    微信公众号 添加方式:
    1:搜索微信号(888888
    2:扫描左侧二维码
  • 快捷导航
    福建二哥 门户 查看主题

    oracle数据库索引失效的问题及解决

    发布者: 网神之王 | 发布时间: 2025-6-18 09:30| 查看数: 90| 评论数: 0|帖子模式

    oracle数据库索引失效问题


    场景

    在开发中有时候遇到某个表中的列明明是创建了索引,但查询时却发现索引失效。

    环境

    下面是工作流activiti中的两张表act_hi_procinst、act_hi_taskinst关系是一对多(一个流程包含多个流程环节),一个是历史流程表,一个是历史流程环节表。



    索引失效情况及验证

    (单表act_hi_procinst已经在delete_reason_列上创建了索引 )

    验证一:索引列为is null 和 is not null时,索引失效
    1. select * from act_hi_procinst t where t.delete_reason_ is not null;
    2. select * from act_hi_procinst t where t.delete_reason_ is null;
    复制代码

    全表扫描,该列索引失效
    1. select * from act_hi_procinst t where t.delete_reason_ ='ACTIVITI_DELETED' and rownum < 1000;
    复制代码

    索引生效,Oracle 数据库使用索引范围扫描方式。
    这种扫描方式通过索引键值的范围来定位需要的数据。
    1. select * from act_hi_procinst t where t.delete_reason_ is not null
    2. and t.start_time_ between TO_DATE('2023-01-01 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    3. and TO_DATE('2023-12-31 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    4. and rownum < 1000
    复制代码


    结论一

    经验证索引列查询使用is null 和 is not null则该列索引失效。
    验证二:索引列为 !=和 <> 时会导致该索引列失效
    1. select * from act_hi_procinst t where t.delete_reason_ !='ACTIVITI_DELETED';
    2. select * from act_hi_procinst t where t.delete_reason_ <>'ACTIVITI_DELETED';
    复制代码


    结论二

    经验证索引列查询使用 !=和 <> 时会导致该索引列失效
    验证三:索引列用函数处理则该索引会失效
    1. select * from act_hi_procinst t where to_char(start_time_,'YYYY')= '2023'
    复制代码


    结论三

    索引列用函数处理则该索引会失效,如字符串函数trunc,to_char,substring,to_date等函数
    区别下面的sql(下面的sql走索引)
    1. select * from act_hi_procinst t where t.start_time_ >= TO_DATE('2023-01-01 00:00:00', 'YYYY-MM-DD HH24:MI:SS');
    复制代码
    验证四:索引列使用like的前置%查询,则该索引列失效。
    1. select * from act_hi_procinst t where t.business_key_ like '%20230103-0000102'
    复制代码

    like 使用后置百分号走索引


    结论四

    经验证索引列使用like的前置%查询时会导致该索引列失效,但是使用了ike的后置%则会走索引
    验证五:范围索引查询和等值索引查询同时存在,则范围索引失效
    其中start_time_创建有普通索引,delete_reason_字段也创建了普通索引

    • SQL一:
    1. select * from act_hi_procinst t where t.start_time_
    2. between TO_DATE('2023-01-01 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    3. and TO_DATE('2023-12-31 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    4. and t.delete_reason_ ='ACTIVITI_DELETED'
    5. and rownum < 1000;
    复制代码

    • SQL二:
    1. select * from act_hi_procinst t
    2. where t.start_time_ >= TO_DATE('2023-01-01 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    3. and t.start_time_<= TO_DATE('2023-12-31 00:00:00', 'YYYY-MM-DD HH24:MI:SS')
    4. and t.delete_reason_ ='ACTIVITI_DELETED'
    5. and rownum < 1000
    复制代码


    结论五

    范围索引查询和等值索引查询同时存在,则范围索引失效
    注:上面的查询sql中加 rownum < 1000的目的主要是数据量太大,这里只是要验证一下查询是否走索引列

    总结

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

    来源:https://www.jb51.net/database/333914nc1.htm
    免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有账号?立即注册

    ×

    最新评论

    QQ Archiver 手机版 小黑屋 福建二哥 ( 闽ICP备2022004717号|闽公网安备35052402000345号 )

    Powered by Discuz! X3.5 © 2001-2023

    快速回复 返回顶部 返回列表