MybatisPlus应用参数类型不一致导致索引失效

业务场景介绍

在电商项目中,有一个商品表【t_goods】和一个商品sku表【t_goods_sku】,具体表结构如下所示:

CREATE TABLE `t_goods`  (
    `id` bigint NOT NULL AUTO_INCREMENT COMMENT '主键id',
    `brand_id` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '品牌ID',
    `buy_count` int NULL DEFAULT 0 COMMENT '购买数量',
    `comment_num` int NULL DEFAULT NULL COMMENT '评论数量',
    `cost` decimal(16, 4) NULL DEFAULT NULL COMMENT '成本价格',
    `goods_name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '商品名称',
    `goods_unit` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '计量单位',
    `grade` decimal(16, 4) NULL DEFAULT NULL COMMENT '商品好评率',
    `intro` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '商品详情',
    ...
    PRIMARY KEY (`id`) USING BTREE,
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '商品表' ROW_FORMAT = Dynamic;

CREATE TABLE `t_goods_sku`  (
    `id` bigint NOT NULL AUTO_INCREMENT COMMENT '主键id',
    `goods_id` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '商品ID',
    `name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '商品名称',
    `unit` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '计量单位',
    `intro` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '商品详情',
    `market_enable` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '上架状态',
    `auth_flag` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '审核状态',
    `price` decimal(16, 4) NULL DEFAULT NULL COMMENT '商品价格',
    `quantity` int NULL DEFAULT NULL COMMENT '库存',
    PRIMARY KEY (`id`) USING BTREE,
    INDEX `idx_goods_id_market_enable_auth_flag`(`goods_id` ASC, `market_enable` ASC, `auth_flag` ASC) USING BTREE COMMENT '商品id、上架状态、审核状态联合索引',
    INDEX `inx_goods_id`(`goods_id` ASC) USING BTREE COMMENT '商品id'
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci COMMENT = '商品sku表' ROW_FORMAT = Dynamic;

在t_goods_sku表中通过goods_id可以找到指定商品的所有sku信息,这正是出现索引失效的场景。

索引失效问题描述

根据goods_id获取该商品对应的sku列表信息,代码如下:

goodsSkuService.list(new LambdaQueryWrapper<GoodsSku>().eq(GoodsSku::getGoodsId, goodsId))

这段代码在实际执行中发现比较慢,通过explain命令查看执行计划:

EXPLAIN
SELECT * FROM qu_goods_sku WHERE goods_id = 99994373;

执行结果如下:
在这里插入图片描述
从执行结果来看,t_goods_sku两个索引,一个都未命中。

索引失效问题分析

MySQL 索引失效是一个常见的性能问题,它可能导致查询变慢甚至全表扫描。以下是一些常见的导致索引失效的情况和解决方法的总结:

  1. 不使用索引列进行查询:当查询条件中不包含索引列,MySQL 将无法使用索引来加速查询。解决方法是确保查询条件中包含适当的索引列,以便 MySQL 可以使用索引来执行查询。

  2. 索引列上使用函数或表达式:如果在索引列上使用函数或表达式,MySQL 将无法使用索引。解决方法是尽量避免在索引列上使用函数或表达式,或者考虑创建函数索引来支持这些查询。

  3. 列类型不匹配:当查询条件中的列类型与索引列的类型不匹配时,MySQL 可能无法使用索引。解决方法是确保查询条件的列类型与索引列的类型相匹配。

  4. 组合索引顺序不正确:对于组合索引,索引列的顺序非常重要。如果查询条件中的列顺序与组合索引的列顺序不匹配,MySQL 可能无法使用索引。解决方法是确保查询条件中的列顺序与组合索引的列顺序一致。

  5. 数据分布不均匀:如果索引列的数据分布不均匀,MySQL 可能无法有效地使用索引。这种情况下,可以考虑重新设计索引或使用分区表来改善数据分布。

  6. 过多的索引:过多的索引可能导致索引失效和性能下降。解决方法是仅创建必要的索引,并定期评估和优化现有索引。

  7. 隐式类型转换:当查询条件中的列类型与索引列的类型不匹配时,MySQL 可能会进行隐式类型转换,导致索引失效。解决方法是确保查询条件中的列类型与索引列的类型完全匹配。

  8. 字符串列使用前缀索引:如果字符串列使用了前缀索引,MySQL 可能无法使用索引。解决方法是考虑增加索引长度或使用全文索引来支持模糊查询。

  9. 高基数列索引失效:当索引列的基数(不同值的数量)非常高时,MySQL 可能会选择不使用索引。解决方法是评估索引列的基数和查询的选择性,并根据情况调整索引策略。

  10. 更新频繁的表索引失效:当表上的索引需要频繁更新时,索引可能会失效。解决方法是根据具体情况权衡索引的更新成本和查询性能,并选择合适的索引策略。

针对笔者项目中碰到的问题,通过分析发现,是由于第3点【列类型不匹配】导致的。

GoodsSku对象的goodsId定义为String类型,但是在使用MybatisPlus进行代码查询对象时LambdaQueryWrapper构建时,可以接受Long型值,所以导致sql中where语句为goods_id = 99994373,然后,goods_sku表中goods_id字段是varchar类型,由于传入的列值与列类型不匹配,导致索引失效。

如何解决呢?只要在代码中将传入的Long类型值转为String即可,代码如下:

goodsSkuService.list(new LambdaQueryWrapper<GoodsSku>().eq(GoodsSku::getGoodsId, String.valueOf(goodsId)))

这样子,执行sql如下:

SELECT * FROM qu_goods_sku WHERE goods_id = '99994373';

使用explain命令查看执行计划:
在这里插入图片描述
从上图可以看出,修改参数值类型之后,命中了索引idx_goods_id_market_enable_auth_flag,实际执行效率也得到了明显提升。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/357662.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java链表(2)

&#x1f435;本篇文章将对双向链表进行讲解&#xff0c;模拟实现双向链表的常用方法 一、什么是双向链表 双向链表在指针域上相较于单链表&#xff0c;每一个节点多了一个指向前驱节点的引用prev以及多了指向最后一个节点的引用last&#xff1a; 二、双向链表的模拟实现 首先…

Python实现时间序列分析AR定阶自回归模型(ar_select_order算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 时间序列分析中&#xff0c;AR定阶自回归模型&#xff08;AR order selection&#xff09;是指确定自回…

GPT栏目:yarn 安装

GPT栏目&#xff1a;yarn 安装 一、前言 在跟GPT交互的时候&#xff0c;发现最近gpt4给出的答案率有了比较明显的提高&#xff0c;简单记录一下&#xff0c;我用gpt4拿到的答案吧。 本人已按照这个步骤成功 二、具体步骤 要安装 yarn&#xff0c;你可以按照以下步骤进行操作…

Ubuntu 22.04.1 LTS 编译安装 nginx-1.22.1,Nginx动静分离、压缩、缓存、黑白名单、跨域、高可用、性能优化

1.Ubuntu 22.04.1 LTS 编译安装 nginx-1.22.1 1.1安装依赖 sudo apt install libgd-dev 1.2下载nginx wget http://nginx.org/download/nginx-1.22.1.tar.gz 1.3解压nginx tar -zvxf nginx-1.22.1.tar.gz 1.4编译安装 cd nginx-1.22.1 编译并指定安装位置&#xff0c;执行安装…

Vue学习笔记之生命周期函数

生命周期示意图如下所示&#xff1a; beforeCreate&#xff1a;组件初始化之前触发该事件created&#xff1a;组件初始化完毕触发该事件beforeMount&#xff1a;Vue应用对象挂载DOM结点之前触发该事件mounted&#xff1a;DOM结点挂载成功之后触发该事件beforeUpdate&#xff1a…

欧拉计划第816题:求大量点的最短距离

本次来解决欧拉计划的第816题: 解: 第一步:最原始的算法 先从简单的情况开始,即原题里的14个点的情况 import mathdef gen_points(n):s = [0] * (2*n)s[0] = 290797for i in range(1, 2*n):s[i] = (s[i - 1] * s[i - 1]) % 50515093p = [(s[2 * i], s[2 * i + 1]) for…

分布式ID是什么,以美团Leaf为例改造融入自己项目【第十一期】

前言 在日常开发中&#xff0c;主键id应用是非常广泛的&#xff0c;但是当涉及到分布式系统的时候&#xff0c;往往需要使用到分布式id&#xff0c;每一个服务里面一套生成规则的不易管理&#xff0c;容易引发冲突。我的IM聊天系统中使用分布式id来生成消息唯一键,为后面幂等做…

OpenHarmony RK3568 启动流程优化

目前rk3568的开机时间有21s&#xff0c;统计的是关机后从按下 power 按键到显示锁屏的时间&#xff0c;当对openharmony的系统进行了裁剪子系统&#xff0c;系统app&#xff0c;禁用部分服务后发现开机时间仅仅提高到了20.94s 优化微乎其微。在对init进程的log进行分析并解决其…

12V-80V车灯芯片都有哪些?-H5028L

电动车车灯芯片的工作原理可以简要概括为以下几点&#xff1a; 光源&#xff1a;电动车车灯通常使用LED&#xff08;Light Emitting Diode&#xff09;作为光源。LED是一种半导体器件&#xff0c;当电流通过LED时&#xff0c;它会发光。 驱动电路&#xff1a;车灯芯片中包含驱…

百度智能小程序开发平台:SEO关键词推广优化 带完整的搭建教程

移动互联网的普及&#xff0c;小程序成为了众多企业和开发者关注的焦点。百度智能小程序开发平台为开发者提供了一站式的解决方案&#xff0c;帮助企业快速搭建并推广自己的小程序。本文将重点介绍百度智能小程序开发平台的SEO关键词推广优化功能&#xff0c;并带完整的搭建教程…

保护医疗数据不受威胁:MPLS专线在医疗网络安全中的角色

随着数字技术的快速发展&#xff0c;医疗行业正在经历一场革命。从电子健康记录到远程医疗服务&#xff0c;数字化不仅提高了效率&#xff0c;也带来了前所未有的挑战--尤其是关于数据安全和隐私保护的挑战。在这样的背景下&#xff0c;如何确保敏感的医疗数据安全传输&#xf…

github添加 SSH 密钥

1 打开终端 输入 ssh-keygen -t rsa -b 4096 -C "github邮箱地址"如果不需要密码可以一路回车 出现这个页面就是生存成功了 open ~/.ssh // 打开.ssh 找到id_rsa.pub复制出内容新建ssh密钥输入内容,保存即可

本地部署Tale博客网站并结合内网穿透实现公网访问本地站点

文章目录 前言1. Tale网站搭建1.1 检查本地环境1.2 部署Tale个人博客系统1.3 启动Tale服务1.4 访问博客地址 2. Linux安装Cpolar内网穿透3. 创建Tale博客公网地址4. 使用公网地址访问Tale 前言 今天给大家带来一款基于 Java 语言的轻量级博客开源项目——Tale&#xff0c;Tale…

数字人解决方案VividTalk——音频驱动单张照片实现人物头像说话的效果

前言 VividTalk是一项由南京大学、阿里巴巴、字节跳动和南开大学共同开发的创新项目。该项目通过结合单张人物静态照片和一段语音录音&#xff0c;能够制作出一个看起来仿佛实际说话的人物视频。项目的特点包括自然的面部表情和头部动作&#xff0c;口型能够同步&#xff0c;同…

C++ 数论相关题目:卡特兰数应用、快速幂求组合数。满足条件的01序列

给定 n 个 0 和 n 个 1 &#xff0c;它们将按照某种顺序排成长度为 2n 的序列&#xff0c;求它们能排列成的所有序列中&#xff0c;能够满足任意前缀序列中 0 的个数都不少于 1 的个数的序列有多少个。 输出的答案对 1097 取模。 输入格式 共一行&#xff0c;包含整数 n 。 …

“全”实力认可 | 美创科技领跑CCSIP 2023全景图数据安全领域

近日&#xff0c;FreeBuf咨询正式发布《CCSIP&#xff08;China Cyber Security Industry Panorama&#xff09;2023中国网络安全行业全景册&#xff08;第六版&#xff09;》。本次全景册面向广大国内安全厂商&#xff0c;由厂商自主申报并填写信息征集表&#xff0c;经FreeBu…

Android 中的动态应用程序图标

Android 中的动态应用程序图标 一、需求二、解决方案三、方案实现四、结论 一、需求 您可能遇到过那些可以实现巧妙技巧的应用程序 - 更改应用程序图标&#xff08;也许是在您的生日那天&#xff09;&#xff0c;然后无缝切换回常规图标。这种功能会激起你的好奇心&#xff0c…

网络防御安全:2-6天笔记

第二章&#xff1a;防火墙 一、什么是防火墙 防火墙的主要职责在于&#xff1a;控制和防护。 防火墙可以根据安全策略来抓取流量之后做出对应的动作。 二、防火墙的发展 区域&#xff1a; Trust 区域&#xff0c;该区域内网络的受信任程度高&#xff0c;通常用来定义内部…

TensorFlow2实战-系列教程9:RNN文本分类1

&#x1f9e1;&#x1f49b;&#x1f49a;TensorFlow2实战-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Jupyter Notebook中进行 本篇文章配套的代码资源已经上传 1、文本分类任务 1.1 文本分类 数据集构建&#xff1a;影评数据集进行情感分析&…

Leetcode 206 反转链表

反转链表 准备工作1&#xff09;ListNode基本结构2&#xff09;初始化ListNode集合 解法一&#xff1a;遍历创建新节点解法二&#xff1a;两组List&#xff0c;面向对象操作解法三&#xff1a;递归调用解法四&#xff1a;直接移动解法五&#xff1a;解法二的面向过程 Leetcode …