一次由 select * 走了索引而引发的思考


问题描述

一次偶然,发现 select * 居然走了索引,看来这是个知识漏洞,想着稍稍研究下。

建表语句如下:

CREATE TABLE `realtime_event_rw` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `device_id` int(10) unsigned NOT NULL COMMENT '接入设备id',
  `time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '事件发生时间',
  `processing_state` tinyint(1) unsigned DEFAULT '2' COMMENT '事件状态:0:已消除,1:已确认,2:未处理,3:暂缓处理 , 4:自动确认,5:漏报',
  `processing_time` timestamp NULL DEFAULT NULL COMMENT '事件处理的时间;已设置为自动获取当前时间戳',
  `video_address` varchar(500) DEFAULT NULL COMMENT '事件视频地址',
  `img_address` varchar(500) DEFAULT NULL COMMENT '图片地址',
  `analysis_img_address` varchar(500) DEFAULT NULL COMMENT '包含检测跟踪信息的图片地址',
  `finger_print` json DEFAULT NULL COMMENT '{"queue_length":0,"free_or_busy":0}',
  `alarm_type` varchar(20) DEFAULT '0' COMMENT '告警方式;格式:0|1|2;0:不告警,1:客户端告警,2:声音告警,3:声光告警,4:手环告警;参见config_dictionary中item_id=115的子节点',
  `alarm_level` smallint(1) unsigned DEFAULT '0' COMMENT '告警级别;0:未知 1:紧急 2:重要 3:一般 4:非关注;\\r\\n\\r\\n参见config_dictionary中item_id=103的子节点',
  `operator_id` int(10) DEFAULT NULL COMMENT '操作人员id',
  `check_user_id` int(10) DEFAULT NULL COMMENT '检查人员id',
  `remark` varchar(500) DEFAULT '' COMMENT '备注信息',
  `ori_img_address` varchar(500) DEFAULT NULL COMMENT '原始图片',
  `warning_num` int(10) DEFAULT '0' COMMENT '危险物品数量',
  `mec_id` varchar(100) NOT NULL COMMENT '盒子告警id',
  `mec_name` varchar(255) DEFAULT NULL COMMENT '盒子名字',
  `mec_event_id` int(10) unsigned NOT NULL COMMENT '盒子事件id',
  `review_state` int(5) DEFAULT '0' COMMENT '数据状态 0上报 1已检测无异常 2已检测有异常',
  `review_warning_num` int(10) DEFAULT NULL COMMENT '复核危险物品数量',
  `receive_time` datetime DEFAULT NULL COMMENT '数据接收时间',
  `review_time` datetime DEFAULT NULL COMMENT '二次处理完成时间',
  PRIMARY KEY (`id`) USING BTREE,
  KEY `idx_rt_event_camera_id` (`device_id`) USING BTREE,
  KEY `idx_rt_event_processing_state` (`processing_state`) USING BTREE,
  KEY `idx_rt_event_time` (`time`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=18 DEFAULT CHARSET=utf8mb4 COMMENT='二次识别告警明细表';

查询语句如下:

select
	*
from
	realtime_event_rw
where
	time between '2022-08-27 17:30:22' and '2022-08-27 19:30:22'

查询结果

在这里插入图片描述

问题分析

用 explain 查看语句执行情况:

在这里插入图片描述

没想到这里使用了索引(一直误以为使用 select * 就会导致索引失效😂)。

select * 走不走索引,关键取决于 where 后面是否包括有效的索引字段,和 select * 没有关系,select * 最大的影响就是额外的 IO 开销。

开始怀疑是因为结果集为空而导致的,于是修改查询条件去查询表中所有数据。

在这里插入图片描述

这种情况 select * 已经不走索引了,所以怀疑可能和结果集数量或者大小有关。因此,尝试只查询两个字段,判断是否和大小结果集有关。

只查询一个字段(主键 id 除外)

在这里插入图片描述

由上图可知,select * 走不走索引与结果集大小无关,而应该和结果集数量有关。

之后,查阅资料后得知:

查询到的结果集数量,如果超过了总行数的 25%,优化器就会认为没有必要走索引了(主键索引除外,仍然会走索引)。

验证结论

场景一:总记录数为 17,结果集记录数为 4,略低于 25%

在这里插入图片描述

场景二:总记录数为 17,结果集记录数为 5,略高于 25%

在这里插入图片描述

经简单验证后,确定结论正确。

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

久违の欢喜

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值