SQL之美第一篇:组合索引巧解错误的BITMAP AND

2023-11-02 17:59

本文主要是介绍SQL之美第一篇:组合索引巧解错误的BITMAP AND,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大家好,我是千寻,一名追求SQL优化极限的开发DBA,欢迎大家一起与我共赏SQL之美妙,如有疑问,欢迎留言,共同研究,下面进入正题:

问题描述:接到开发反馈,有一个SQL查询不出来结果,原因是字段【pysts_cd】是varchar2类型,去掉符号‘’ 就可以查出来结果,确认数据库没有问题数据以及各种异常数据(比如null,空值,特殊符号,都不存在)

原SQL:

select *

  from qx_prd_psch_d m

 where m.bqx_flag = 'spddqx'

   and m.is_sq_bqx = '0'

   and m.is_del = 0

   and m.pysts_cd ='2'

   and to_char(m.PAYSE_DT, 'yyyy-mm-dd') = '2019-04-17';

拿到SQL,直接执行确认开发说法无误,观察带上符号‘’于不带的两种执行计划:

可以看到,这里走了位图索引转换,同时两个索引做了合并,谓词条件里面的字段都走了索引,再看看出来结果的SQL执行计划:

对比发现了,RS_QX_PRD_PSCH_D_05这个索引一走就出现问题,单独走索引RS_QX_PRD_PSCH_D_04就没有问题,下面来做验证:

这里使用了to_single_byte函数做了全角转半角,当然,trim也可以,

果然,结果出来了。看到这里,部分同学会想那问题不就解决了,加个函数转换字段【pysts_cd】即可,肯定是还存在问题数据(PS:一般情况,这么解释就可以交任务了)当然没那么简单,哈哈。

开发反馈,这个SQL改动涉及代码很多,工作量太大,不给改,希望在数据库后台就解决了,因此,这里开始深入分析:

首先,确认问题字段类型:

图片

第二步,观察相关索引:

图片

这里稍微总结下:谓词条件一共使用了【   bqx_flag,is_sq_bqx,is_del,pysts_cd,PAYSE_DT】5个字段,那么问题出在哪里呢?有经验的人已经发现了 索引【RS_QX_PRD_PSCH_D_05】一定有问题!

第三步,测试索引问题所在

首先,问题所在pysts_cd字段肯定不能动,那么去掉该组合索引另一个字段试试呢?

结果出来了!同样的执行计划,同样的 Plan Hash Value,为什么会出现这种情况呢?细心的小伙伴已经发现啦,没错 你的想法是正确的!

就是谓词条件的filter里面字段的问题,多说一嘴,

Access:表示对应的谓词条件会影响数据的访问路径(是按照索引还是表)

Filter:表示谓词条件只会起到过滤作用,不会影响数据的访问路径。

难道是问题数据?很多人会被误导:

正好排序第一位是1,跟0不匹配,然后就被filter把后面的全都过滤了,让这部分索引结果是为空,导致两个索引走bitmap and以后结果集是空 ,这个解释也说得通,然而事实并非如此,让我们来修改再看看:

到这里,基本绝大部分人思路已经断了,嘿嘿,下面如何排查呢?

这里,就需要抓取真实执行计划了:

瞬间,问题清晰!【SYS_NC00043$】这个对象一般是oracle处理字符乱码才会出现的,查看IS_DEL类型

图片

,发现ORACLE小BUG,位图索引转换NUMBER(1)类型字段的时候,转换的值是乱码!这里又有BITMAP AND 导致连锁反应,结果集为0!

再次小心求证:

确认无误,问题字段其实并不是【pysts_cd】,反而是【is_del】!

问题明确了,解决方案很简单:

  1. 干掉索引 【RS_qx_prd_psch_d_04】+【RS_qx_prd_psch_d_05】

  2. 重建新的组合索引:

图片

既去掉了多个索引的冗余,又提高了SQL的效率,同时解决了开发人员的需求,有效降低他的工作量,收获了开发的崇拜之心,沉浸在SQL美妙的海洋,开心!

优秀的DBA,就该这样!

这篇关于SQL之美第一篇:组合索引巧解错误的BITMAP AND的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/332311

相关文章

MySQL 中的 JSON 查询案例详解

《MySQL中的JSON查询案例详解》:本文主要介绍MySQL的JSON查询的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 的 jsON 路径格式基本结构路径组件详解特殊语法元素实际示例简单路径复杂路径简写操作符注意MySQL 的 J

Windows 上如果忘记了 MySQL 密码 重置密码的两种方法

《Windows上如果忘记了MySQL密码重置密码的两种方法》:本文主要介绍Windows上如果忘记了MySQL密码重置密码的两种方法,本文通过两种方法结合实例代码给大家介绍的非常详细,感... 目录方法 1:以跳过权限验证模式启动 mysql 并重置密码方法 2:使用 my.ini 文件的临时配置在 Wi

MySQL重复数据处理的七种高效方法

《MySQL重复数据处理的七种高效方法》你是不是也曾遇到过这样的烦恼:明明系统测试时一切正常,上线后却频频出现重复数据,大批量导数据时,总有那么几条不听话的记录导致整个事务莫名回滚,今天,我就跟大家分... 目录1. 重复数据插入问题分析1.1 问题本质1.2 常见场景图2. 基础解决方案:使用异常捕获3.

SQL中redo log 刷⼊磁盘的常见方法

《SQL中redolog刷⼊磁盘的常见方法》本文主要介绍了SQL中redolog刷⼊磁盘的常见方法,将redolog刷入磁盘的方法确保了数据的持久性和一致性,下面就来具体介绍一下,感兴趣的可以了解... 目录Redo Log 刷入磁盘的方法Redo Log 刷入磁盘的过程代码示例(伪代码)在数据库系统中,r

mysql中的group by高级用法

《mysql中的groupby高级用法》MySQL中的GROUPBY是数据聚合分析的核心功能,主要用于将结果集按指定列分组,并结合聚合函数进行统计计算,下面给大家介绍mysql中的groupby用法... 目录一、基本语法与核心功能二、基础用法示例1. 单列分组统计2. 多列组合分组3. 与WHERE结合使

Mysql用户授权(GRANT)语法及示例解读

《Mysql用户授权(GRANT)语法及示例解读》:本文主要介绍Mysql用户授权(GRANT)语法及示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql用户授权(GRANT)语法授予用户权限语法GRANT语句中的<权限类型>的使用WITH GRANT

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Mysql如何解决死锁问题

《Mysql如何解决死锁问题》:本文主要介绍Mysql如何解决死锁问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录【一】mysql中锁分类和加锁情况【1】按锁的粒度分类全局锁表级锁行级锁【2】按锁的模式分类【二】加锁方式的影响因素【三】Mysql的死锁情况【1

Windows Docker端口占用错误及解决方案总结

《WindowsDocker端口占用错误及解决方案总结》在Windows环境下使用Docker容器时,端口占用错误是开发和运维中常见且棘手的问题,本文将深入剖析该问题的成因,介绍如何通过查看端口分配... 目录引言Windows docker 端口占用错误及解决方案汇总端口冲突形成原因解析诊断当前端口情况解

SQL BETWEEN 的常见用法小结

《SQLBETWEEN的常见用法小结》BETWEEN操作符是SQL中非常有用的工具,它允许你快速选取某个范围内的值,本文给大家介绍SQLBETWEEN的常见用法,感兴趣的朋友一起看看吧... 在SQL中,BETWEEN是一个操作符,用于选取介于两个值之间的数据。它包含这两个边界值。BETWEEN操作符常用