MySQL请求使用JSON索引查询数据量不准确

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介: MySQL请求使用JSON索引查询数据量不准确

背景描述


通过SQL 语句查询,使用JSON索引的情况下,查询结果不准确,远远大于预期。

如查询SQL:

SELECTcount(1)FROMwheretime='2022-8-22';

结果:3981392

实际结果应该是100W左右


知识点


Json 类型简单介绍


有一种叫做JSON (JavaScript Object Notation) 的轻量级数据交换格式能够替代XML的工作。它就是JSON。

数据格式比较简单, 易于读写, 格式都是压缩的, 占用带宽小,易于解析这种语言。

示例:

json = {"name1":"test", "list":["a", "d", "c"]}


列表数据类型


“列表”是一个值,它包含多个字构成的序列。 “列表值”指的是列表本身,而不是指列表中的值。 列表中的值称为表项,表项用逗号隔开。

示例:

list= ["a", "b", "c"]


组合索引数据结构

  1. 如图所示,下面是两条记录。
  2. 若通过全表扫描,将返回2条记录。
  3. 若通过日期字段与JSON 字段,组合索引进行扫描,这时日期字段将与json 列表中每一个元素一一匹配,所以将返回6条记录。


复现业务场景

  1. 创建表
  2. 插入准备数据
  3. 通过全表扫描,得到真实的表中记录条目
  4. 通过json 类型的组合索引扫描,得到json 列表元素组合的条目。
  5. 通过page 的数据结构( information_schema.INNODB_BUFFER_PAGE),确定页中记录的条目数量。

1. 创建表

CREATETABLE `t1` (  `MoveTime` datetimeNOTNULL,  `NodeTree` json DEFAULT NULL,  `SaleCount` bigintNOTNULL DEFAULT '0',  KEY `Idx_MoveTime_SalueCount_Tree1` (`MoveTime`,(cast(json_extract(`NodeTree`,_utf8mb4'$.node')aschar(32) array)),`SaleCount`) USING BTREE
) ENGINE=InnoDB;

2. 准备数据

INSERTINTO t1(`MoveTime`, `NodeTree`, `SaleCount`)VALUES('2022-08-22 00:00:00','{"node": ["393459011", "industrial", "12900351", "12899801", "office-products", "1069242"]}','549');INSERTINTO t1(`MoveTime`, `NodeTree`, `SaleCount` )VALUES('2022-08-22 00:00:00','{"node": ["166099011", "166092011", "toys-and-games"]}','11978');INSERTINTO t1(`MoveTime`, `NodeTree`, `SaleCount`)VALUES('2022-08-22 00:00:00','{"node": ["1069462", "1069454", "1069242", "office-products", "490790011", "12899801"]}','2972');

3. 验证


  1. 通过全表扫描验证
selectcount(1)from t1 where MoveTime ='2022-08-22';explain selectcount(1)from t1 where MoveTime ='2022-08-22';

通过结果,可以看到通过全表扫描查看到的结果是按照表中的记录数进行统计。


  1. 通过带有 json 数据类型索引进行验证
selectcount(1)from t1 force index(Idx_MoveTime_SalueCount_Tree1)where MoveTime ='2022-08-22';explain selectcount(1)from t1 force index(Idx_MoveTime_SalueCount_Tree1)where MoveTime ='2022-08-22';



4. 查看数据页 page 中的记录条目数量

select table_name,index_name,number_records,data_size from information_schema.INNODB_BUFFER_PAGEwhere table_name like'`db01`.`t1`';



5. 结论

当json 数据为列表时,使用联合索引会,其它字段与列表匹配,会产生一对多的关系。从而最终统计数量就会按符合列表的数量统计。通过查看 information_schema.INNODB_BUFFER_PAGE ,对应的索引的记录数,可以确定,低层数据结构就是这么设计的。



解决方案

  1. 建议合理使用 json 索引。
  2. 可以针对不同的统计信息,指定不同的索引,进行统计。



适用版本

适用MySQL 5.7以上版本

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
5天前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
本文介绍了MySQL InnoDB存储引擎中的数据文件和重做日志文件。数据文件包括`.ibd`和`ibdata`文件,用于存放InnoDB数据和索引。重做日志文件(redo log)确保数据的可靠性和事务的持久性,其大小和路径可由相关参数配置。文章还提供了视频讲解和示例代码。
105 11
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
|
5天前
|
缓存 NoSQL 关系型数据库
Redis和Mysql如何保证数据⼀致?
在项目中,为了解决Redis与Mysql的数据一致性问题,我们采用了多种策略:对于低一致性要求的数据,不做特别处理;时效性数据通过设置缓存过期时间来减少不一致风险;高一致性但时效性要求不高的数据,利用MQ异步同步确保最终一致性;而对一致性和时效性都有高要求的数据,则采用分布式事务(如Seata TCC模式)来保障。
33 14
|
8天前
|
SQL 前端开发 关系型数据库
SpringBoot使用mysql查询昨天、今天、过去一周、过去半年、过去一年数据
SpringBoot使用mysql查询昨天、今天、过去一周、过去半年、过去一年数据
37 9
|
10天前
|
缓存 监控 关系型数据库
如何优化MySQL查询速度?
如何优化MySQL查询速度?【10月更文挑战第31天】
33 3
|
13天前
|
监控 关系型数据库 MySQL
数据库优化:MySQL索引策略与查询性能调优实战
【10月更文挑战第27天】本文深入探讨了MySQL的索引策略和查询性能调优技巧。通过介绍B-Tree索引、哈希索引和全文索引等不同类型,以及如何创建和维护索引,结合实战案例分析查询执行计划,帮助读者掌握提升查询性能的方法。定期优化索引和调整查询语句是提高数据库性能的关键。
70 1
|
14天前
|
SQL 监控 关系型数据库
MySQL如何查看每个分区的数据量
通过本文的介绍,您可以使用MySQL的 `INFORMATION_SCHEMA`查询每个分区的数据量。了解分区数据量对数据库优化和管理具有重要意义,可以帮助您优化查询性能、平衡数据负载和监控数据库健康状况。希望本文对您在MySQL分区管理和性能优化方面有所帮助。
25 1
|
14天前
|
监控 关系型数据库 MySQL
数据库优化:MySQL索引策略与查询性能调优实战
【10月更文挑战第26天】数据库作为现代应用系统的核心组件,其性能优化至关重要。本文主要探讨MySQL的索引策略与查询性能调优。通过合理创建索引(如B-Tree、复合索引)和优化查询语句(如使用EXPLAIN、优化分页查询),可以显著提升数据库的响应速度和稳定性。实践中还需定期审查慢查询日志,持续优化性能。
45 0
|
分布式计算 关系型数据库 MySQL
E-Mapreduce如何处理RDS的数据
目前网站的一些业务数据存在了数据库中,这些数据往往需要做进一步的分析,如:需要跟一些日志数据关联分析,或者需要进行一些如机器学习的分析。在阿里云上,目前E-Mapreduce可以满足这类进一步分析的需求。
4971 0
|
7天前
|
SQL 关系型数据库 MySQL
go语言数据库中mysql驱动安装
【11月更文挑战第2天】
20 4
|
4天前
|
SQL 关系型数据库 MySQL
12 PHP配置数据库MySQL
路老师分享了PHP操作MySQL数据库的方法,包括安装并连接MySQL服务器、选择数据库、执行SQL语句(如插入、更新、删除和查询),以及将结果集返回到数组。通过具体示例代码,详细介绍了每一步的操作流程,帮助读者快速入门PHP与MySQL的交互。
13 1