首页 游戏 软件 资讯 排行榜 专题
首页
数据库
SQL索引列使用函数导致失效的原因与优化方法

SQL索引列使用函数导致失效的原因与优化方法

热心网友
40
转载
2026-05-07

为什么在SQL查询中不建议对索引列使用函数

为什么在SQL查询中不建议对索引列使用函数_避免索引失效的优化策略

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

在数据库查询优化中,有一条几乎被奉为金科玉律的原则:尽量避免在WHERE子句中对索引列使用函数。无论是DATE()SUBSTR()还是YEAR(),这类操作都极有可能让精心设计的索引瞬间“哑火”。这背后的原因,并非数据库优化器不够智能,而是源于B+树索引最底层的存储与匹配逻辑。

函数使索引失效是因为B+树索引存储原始值,而函数运算后无法匹配有序结构;应将函数移至常量侧,如DATE(create_time)='2024-01-01'改为create_time>='2024-01-01' AND create_time<'2024-01-02'。

为什么函数会让索引失效

要理解这一点,得先看看索引是怎么工作的。想象一下,索引就像一本按字母顺序排列的电话簿。如果有一列create_time,其索引存储的就是类似‘2024-01-01 14:23:05’这样的原始时间戳,并且严格按照这个完整的时间顺序排列。

一旦你写下WHERE DATE(create_time) = ‘2024-01-01’这样的查询,情况就变了。数据库优化器会犯难:它无法直接利用这本按“完整时间戳”排序的电话簿,去快速查找所有“日期部分等于某一天”的记录。它被迫退回到最原始的方法——翻开每一页(即每一行数据),先调用DATE()函数取出日期部分,再进行比较。这本质上就是一次全表扫描。

所以说,核心问题在于:优化器将“对索引列施加了不可下推的运算”视为一个信号,它无法将这种计算后的条件,映射回索引原有的有序结构上。索引的快速定位能力,就此被绕过了。

DATE()YEAR() 等日期函数怎么改写

那么,正确的姿势是什么?核心原则其实很清晰:让索引列以“裸值”的形式出现在比较操作中,而把所有的函数或计算都挪到等式的常量一侧。

  • 错误写法:WHERE DATE(create_time) = ‘2024-01-01’
  • 优化写法:WHERE create_time >= ‘2024-01-01’ AND create_time < ‘2024-01-02’

同理:

  • WHERE YEAR(create_time) = 2024 应改为 WHERE create_time >= ‘2024-01-01’ AND create_time < ‘2025-01-01’
  • 在PostgreSQL中,类似WHERE create_time::date = ‘2024-01-01’的类型转换操作也应避免,同样改用范围查询。

这里有个小提示:虽然BETWEEN也能实现范围查询,但它容易在边界精度上出问题(例如BETWEEN ‘2024-01-01’ AND ‘2024-01-01 23:59:59’)。相比之下,使用左闭右开区间(>= 配合 <)不仅更安全,逻辑上也更清晰。

字符串函数如 SUBSTR()UPPER() 怎么处理

日期函数的问题,在字符串函数上同样存在。像SUBSTR(name, 1, 3) = ‘adm’这样的查询,会破坏索引基于前缀的匹配能力,导致其无法被使用。

面对这类情况,可以按以下思路解决:

  • 优先考虑前缀匹配:如果能转化为前缀查询,那是最理想的。例如,将上述查询改为WHERE name LIKE ‘adm%’,这样就能充分利用name列上的普通B+树索引。
  • 从源头设计索引:如果业务上经常需要进行大小写不敏感的匹配,与其在查询时使用UPPER(name),不如在创建表或索引时,就为字符列指定不区分大小写的校对规则(例如MySQL的utf8mb4_0900_as_ci)。
  • 慎用函数索引:对于某些必须依赖函数结果且查询非常频繁的场景,可以考虑创建函数索引(如PostgreSQL/Oracle支持CREATE INDEX idx_name ON t(UPPER(name)))。但这是一把双刃剑,它会增加索引维护的开销,影响写入性能,切忌滥用。

容易被忽略的隐含陷阱

除了显而易见的函数调用,还有一些“看起来人畜无害”的写法,其实也暗藏杀机。它们本质上也是对列进行了计算或转换:

  • 算术运算:WHERE id + 0 = 100 —— 索引列参与了计算,索引失效。
  • 表达式操作:WHERE username || ‘’ = ‘alice’(PostgreSQL中的字符串拼接)—— 同样是表达式操作,索引无法生效。
  • 函数在常量侧:WHERE create_time = CURDATE() —— 注意,CURDATE()本身就是一个函数(等同于DATE(NOW()))。虽然函数作用在常量侧,但某些情况下优化器可能无法准确估算,稳妥起见应改为WHERE create_time >= CURDATE() AND create_time < CURDATE() + INTERVAL 1 DAY

最后,给出一个最稳妥的建议:不要凭感觉猜测。任何不确定的查询优化,都应该通过执行EXPLAIN命令来验证。关键要看执行计划中的key字段是否非空(表示使用了索引),以及type字段是否达到了ref或更好的级别(如eq_refconst)。数据库优化器的实际行为,远比我们“看起来差不多”的直觉要复杂。

来源:https://www.php.cn/faq/2420650.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

车载吸尘器滤网拆卸方法 如何解开卡扣设计
电脑教程
车载吸尘器滤网拆卸方法 如何解开卡扣设计

是的,卡扣式滤网是主流车载无线吸尘器的标配 打开市面上任何一款主流车载吸尘器,你会发现,前盖滤网几乎清一色采用了卡扣式结构。这可不是偶然。这种设计通过精密匹配的旋转卡扣,真正实现了“秒拆秒装”——用户单手轻拧大约90度,前盖应声而开,多层复合滤网便呈现在眼前。滤网本身通常由可水洗的HEPA层和初效海

热心网友
05.07
雷神笔记本UEFI启动U盘制作与BIOS设置教程
电脑教程
雷神笔记本UEFI启动U盘制作与BIOS设置教程

雷神笔记本实现UEFI模式U盘启动,核心在于正确配置BIOS中的安全启动与UEFI引导选项,并确保U盘启动介质符合UEFI规范。 具体操作时,得先插入那个已经准备好的、符合UEFI规范的启动U盘。开机一瞬间,手速要快,连续按F12进入启动菜单。如果够顺利,你会直接看到一个带有“UEFI: [你的U盘

热心网友
05.07
车载吸尘器滤网清洗指南 水洗的正确方法与注意事项
电脑教程
车载吸尘器滤网清洗指南 水洗的正确方法与注意事项

车载吸尘器滤网能否水洗,关键在这儿 很多车主都纠结过这个问题:吸尘器滤网脏了,到底能不能用水洗?答案其实不复杂,核心就两点——看材质,看设计。不是所有的滤网都经得起“洗礼”,也不是所有号称能洗的滤网都一个洗法。根据海尔、德尔玛这些主流品牌的官方指南和业内清洁经验,这事儿有明确的“安全区”和“禁区”:

热心网友
05.07
vivo Y31手机联系人备份导出详细步骤
电脑教程
vivo Y31手机联系人备份导出详细步骤

vivo Y31联系人备份:最便捷高效的本地导出指南 想把vivo Y31里的通讯录完整备份下来,以备不时之需?最省心、兼容性最强的方法,莫过于利用手机自带的“联系人”应用,直接导出为通用的vCard ( vcf) 文件。整个过程不需要你安装任何第三方软件,也无需登录云端账号,几步操作就能在手机存储

热心网友
05.07
雷蛇鼠标灵敏度快速调节设置方法
电脑教程
雷蛇鼠标灵敏度快速调节设置方法

雷蛇鼠标调灵敏度最快的方式,是直接按压机身自带的物理DPI切换键 要说最直接、最快的方式,那绝对是机身上那个物理DPI切换键。它最大的好处,是彻底绕开了软件、系统和网络延迟——手指按下去,灵敏度瞬间切换,整个过程在毫秒间完成,真正实现了“所想即所得”。像Razer DeathAdder V3和Bas

热心网友
05.07

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

Bitget交易所2026年发展前景与市场排名深度解析
web3.0
Bitget交易所2026年发展前景与市场排名深度解析

2026年,Bitget在交易所排行榜上展现出强劲的竞争力。其表现主要体现在用户资产安全体系的持续加固、多元化产品矩阵的成熟与创新,以及在合规与全球化布局上的显著进展。平台通过优化现货与衍生品交易体验,并深化Web3生态建设,巩固了其在行业中的领先地位,获得了市场与用户的广泛认可。

热心网友
05.07
NET开发中HttpClient使用避坑指南与最佳实践详解
编程语言
NET开发中HttpClient使用避坑指南与最佳实践详解

HttpClient的7个常见陷阱与规避指南 在 NET 生态里进行项目开发,HttpClient 几乎是调用外部 API 绕不开的一个工具。它的上手门槛很低,用起来很顺手,但恰恰是这份“简单”,让不少开发者放松了警惕。如果不清楚它内部的运作机制,一不小心就可能掉进坑里,轻则请求失败,重则引发服务

热心网友
05.07
NETCore与Linux服务器时间同步问题的多种解决方案详解
编程语言
NETCore与Linux服务器时间同步问题的多种解决方案详解

如何解决 NET Core项目与Linux服务器之间的时间同步问题 导语 搞分布式系统的开发者,多少都踩过时间不同步的“坑”。这事说大不大,说小不小——日志对不上、订单乱取消、交易出岔子,追根溯源,往往是几台机器的时间“各走各的”。尤其是在 NET Core应用遇上Linux服务器的场景,时区、格式

热心网友
05.07
NET 4.7 如何使用 NLog 将日志记录到数据库
编程语言
NET 4.7 如何使用 NLog 将日志记录到数据库

1 首先安装必要的NuGet包 第一步,咱们得把项目里需要的“砖瓦”——也就是那几个关键的NuGet包——给准备好。具体是下面这几个: NLog:日志记录的核心库。 NLog Config (可选):如果你想让配置文件自动生成,可以加上这个。 当然,别忘了根据你用的数据库类型,安装对应的提供程序。

热心网友
05.07
NETCore消息队列RabbitMQ实现方法与代码示例
编程语言
NETCore消息队列RabbitMQ实现方法与代码示例

在 NET Core 中玩转 RabbitMQ:从零搭建可靠的消息队列 消息队列是现代应用解耦和异步通信的基石,而 RabbitMQ 无疑是这个领域的明星选手。它基于 AMQP 协议,为不同应用程序间的可靠消息传递提供了强大支持。今天,我们就来深入聊聊,如何在 NET Core 环境中,亲手搭建

热心网友
05.07