MySQL分库分表多维度查询——全表冗余

2023-10-12 17:30

本文主要是介绍MySQL分库分表多维度查询——全表冗余,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

分表分库面临的问题

MySQL分库分表,一般只能按照一个维度进行查询.
以订单表为例, 按照用户ID mod 64 分成 64个数据库.
按照用户的维度查询很快,因为最终的查询落在一台服务器上.

非分区逻辑字段查询

但是如果按照商户的维度查询,则代价非常高.
需要查询全部64台服务器.

分页查询

在分页的情况下,更加恶化.
比如某个商户查询第10页的数据(按照订单的创建时间).需要在每台数据库服务器上查询前100条数据,程序收到 64*100 条数据,然后按照订单的创建时间排序,截取排名90-100号的10条记录返回,然后抛弃其余的6390条记录.如果查询的是第100页,第1000页,则对数据库IO,网络,中间件CPU,都是不小的压力.

常见解决方案

分库分表之后,为了应对多维度查询,很多情况下会引入冗余.

  • 全表冗余

全表冗余

比如两个集群,一个按照用户ID分库分表,另外一个按照商户ID分库分表.
这样多维度查询的时候,各查各的.
但是有几个问题,一样不好解决.
比如,
每扩展一个维度,就需要引入一个集群.
集群间的数据,如何保证一致性.
冗余占用大量磁盘空间.
从朋友那里看到的订单表结构.做冗余会占用大量的磁盘空间.

用户ID分表分库和商户ID分表分库实际操作

关键问题:保持两份数据的一致性。
通过消息中间件保持事务操作,然后分表进行入库操作,保持最终一致性。

案例
create table TS_ORDER  
(  ORDER_ID        NUMBER(8) not null,  SN              VARCHAR2(50),  MEMBER_ID       NUMBER(8),  STATUS          NUMBER(2),  PAY_STATUS      NUMBER(2),  SHIP_STATUS     NUMBER(2),  SHIPPING_ID     NUMBER(8),  SHIPPING_TYPE   VARCHAR2(255),  SHIPPING_AREA   VARCHAR2(255),  PAYMENT_ID      NUMBER(8),  PAYMENT_NAME    VARCHAR2(50),  PAYMENT_TYPE    VARCHAR2(50),  PAYMONEY        NUMBER(20,2),  CREATE_TIME     NUMBER(20) not null,  SHIP_NAME       VARCHAR2(255),  SHIP_ADDR       VARCHAR2(255),  SHIP_ZIP        VARCHAR2(20),  SHIP_EMAIL      VARCHAR2(50),  SHIP_MOBILE     VARCHAR2(50),  SHIP_TEL        VARCHAR2(50),  SHIP_DAY        VARCHAR2(255),  SHIP_TIME       VARCHAR2(255),  IS_PROTECT      VARCHAR2(1),  PROTECT_PRICE   NUMBER(20,2),  GOODS_AMOUNT    NUMBER(20,2),  SHIPPING_AMOUNT NUMBER(20,2),  ORDER_AMOUNT    NUMBER(20,2),  WEIGHT          NUMBER(20,2),  GOODS_NUM       NUMBER(8),  GAINEDPOINT     NUMBER(11) default 0,  CONSUMEPOINT    NUMBER(11) default 0,  DISABLED        VARCHAR2(1),  DISCOUNT        NUMBER(20,2),  IMPORTED        NUMBER(2) default 0,  PIMPORTED       NUMBER(2) default 0,  COMPLETE_TIME   NUMBER(11) default 0,  CANCEL_REASON   VARCHAR2(255),  SIGNING_TIME    NUMBER(11),  THE_SIGN        VARCHAR2(255),  ALLOCATION_TIME NUMBER(11),  SHIP_PROVINCEID NUMBER(11),  SHIP_CITYID     NUMBER(11),  SHIP_REGIONID   NUMBER(11),  SALE_CMPL       NUMBER(2),  SALE_CMPL_TIME  NUMBER(11),  DEPOTID         NUMBER(11),  ADMIN_REMARK    VARCHAR2(1000),  COMPANY_CODE    VARCHAR2(32),  PARENT_SN       VARCHAR2(50),  REMARK          VARCHAR2(100),  GOODS           CLOB,  ORIGINAL_AMOUNT NUMBER(20,2),  IS_ONLINE       CHAR(1),  IS_COMMENTED    CHAR(1) default 0,  ORDER_FLAG      CHAR(1) default 1  
)  

可以试试用表代替索引的方法.
1.分库分表
2.最终一致性
3.用表代替索引的功能
在这里插入图片描述
首先,还是基于分库分表.订单表按照用户ID mod 64 分到不同的服务器上(按照查询最多的维度分)。

数据库服务器1 的数据库名称为 db_1
数据库服务器2 的数据库名称为 db_2

以db_1为例,创建如下表
1.订单表
TS_ORDER_1 分区表,每个月一个分区.

2.事务表
create table tran_log_1(
tran_id bigint primary key,
param varchar(2000)
);
分区表,每个月一个分区.

3.消息表
create table msg_log_1(
tran_id bigint,
shardKey varchar(20) not null,
primary key(tran_id,shardKey)
);
分区表,每个月一个分区.

4.维度索引表
create table shard_shop_1(
id bigint primary key auto_increment,
shopid int,
ts timestamp,
state int,
dbid int,
orderid bigint,
index(shopid,ts,state)
);
分区表,每个月一个分区.

关于使用事务表,消息表实现分库分表最终一致性请参考
http://blog.itpub.net/29254281/viewspace-1819422/

关于集群主键生成服务请参考
http://blog.itpub.net/29254281/viewspace-1811711/

订单创建的流程
Web服务器接收到用户订单,首先通过RPC获取一个事务ID(tran_id).
用事务ID mod 64 找到数据库服务器,
将事务ID,参数写入tran_log 表,
然后将事务ID,参数写入消息队列.
如果写入消息队列成功,则提交事务.否则回滚事务.
此时就可以返回用户界面.

后端处理服务收到消息队列的信息,首先查询tran_log 表,是否存在这个事务ID,如果不存在则不予处理.
然后将队列的消息,分为两个维度分别处理,一个是用户维度,一个是商户维度.
作为用户维度,
先根据用户ID mod 64 找到最终落地的数据库,查询那个数据库的消息表msg_log,在用户维度,是否存在这个事务ID,如果存在,则不予处理.
(select count(*) from msg_log_XX where shardKey=‘订单创建:用户维度’ and tran_id=?)
如果不存在,则开启一个事务
插入订单表,我觉得可以用tran_id直接作为订单的ID,
并且插入消息表 insert msg_log_XX(tran_id,shardKey) values(?,‘订单创建:用户维度’);
提交事务,commit.

作为商户维度,
则根据商户ID mod 64 找到最终的数据库,和用户维度的数据库,可能不是同一台服务器.
同样,也是先查询落地数据库的消息表,
(select count(*) from msg_log_XXX where shardKey=‘订单创建:商户维度’ and tran_id=?)
如果不存在记录,则开启事务,
插入维度索引表,
insert into shard_shop_XXX(shopid,ts,state,dbid,orderid) values(…)
shopid,ts,state 商户ID,订单时间,订单状态都是根据订单的原始信息.
dbid 指的是 根据用户维度(主维度),订单数据所在的数据库ID,
orderid 指的是 在用户维度(主维度),订单表的主键.

插入消息表,insert msg_log_XX(tran_id,shardKey) values(?,‘订单创建:商户维度’);
最后提交.

这样,作为商户维度查询的时候,先根据商户的ID mod 64 找到 维度索引表,获取该商户的订单信息
select * from shard_shop_1 where shopid=? and state=2 order by ts limit 300,10;
获取的信息可能如下
在这里插入图片描述
可以看到,符合条件的订单信息,分别来自 服务器1,2,16,32,64
获取了这部分信息,就可以直接去这些服务器上取数据,并且是主键查询,速度很快.

每隔一段时间,由后台程序,查看 tran_log和msg_log,如果发现有缺失的数据,则进行事务补偿.

扩展的时候,则新增维度索引表即可.

因为所有的表,都是按月的分区表,可以将过去的冷数据,在一个服务器集中存放,这个实例就同时存放64个数据库.毕竟都是冷数据,访问量很小.
能分还要能合.比如:

参考文章:
http://blog.itpub.net/29254281/viewspace-2086198/

这篇关于MySQL分库分表多维度查询——全表冗余的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Mybatis对MySQL if 函数的不支持问题解读

《Mybatis对MySQLif函数的不支持问题解读》接手项目后,为了实现多租户功能,引入了Mybatis-plus,发现之前运行正常的SQL语句报错,原因是Mybatis不支持MySQL的if函... 目录MyBATis对mysql if 函数的不支持问题描述经过查询网上搜索资料找到原因解决方案总结Myb

MySQL 筛选条件放 ON后 vs 放 WHERE 后的区别解析

《MySQL筛选条件放ON后vs放WHERE后的区别解析》文章解释了在MySQL中,将筛选条件放在ON和WHERE中的区别,文章通过几个场景说明了ON和WHERE的区别,并总结了ON用于关... 今天我们来讲讲数据库筛选条件放 ON 后和放 WHERE 后的区别。ON 决定如何 "连接" 表,WHERE

mysql_mcp_server部署及应用实践案例

《mysql_mcp_server部署及应用实践案例》文章介绍了在CentOS7.5环境下部署MySQL_mcp_server的步骤,包括服务安装、配置和启动,还提供了一个基于Dify工作流的应用案例... 目录mysql_mcp_server部署及应用案例1. 服务安装1.1. 下载源码1.2. 创建独立

Mysql中RelayLog中继日志的使用

《Mysql中RelayLog中继日志的使用》MySQLRelayLog中继日志是主从复制架构中的核心组件,负责将从主库获取的Binlog事件暂存并应用到从库,本文就来详细的介绍一下RelayLog中... 目录一、什么是 Relay Log(中继日志)二、Relay Log 的工作流程三、Relay Lo

MySQL日志UndoLog的作用

《MySQL日志UndoLog的作用》UndoLog是InnoDB用于事务回滚和MVCC的重要机制,本文主要介绍了MySQL日志UndoLog的作用,文中介绍的非常详细,对大家的学习或者工作具有一定的... 目录一、Undo Log 的作用二、Undo Log 的分类三、Undo Log 的存储四、Undo

mybatis-plus分表实现案例(附示例代码)

《mybatis-plus分表实现案例(附示例代码)》MyBatis-Plus是一个MyBatis的增强工具,在MyBatis的基础上只做增强不做改变,为简化开发、提高效率而生,:本文主要介绍my... 目录文档说明数据库水平分表思路1. 为什么要水平分表2. 核心设计要点3.基于数据库水平分表注意事项示例

MySQL游标和触发器的操作流程

《MySQL游标和触发器的操作流程》本文介绍了MySQL中的游标和触发器的使用方法,游标可以对查询结果集进行逐行处理,而触发器则可以在数据表发生更改时自动执行预定义的操作,感兴趣的朋友跟随小编一起看看... 目录游标游标的操作流程1. 定义游标2.打开游标3.利用游标检索数据4.关闭游标例题触发器触发器的基

MySQL查看表的历史SQL的几种实现方法

《MySQL查看表的历史SQL的几种实现方法》:本文主要介绍多种查看MySQL表历史SQL的方法,包括通用查询日志、慢查询日志、performance_schema、binlog、第三方工具等,并... 目录mysql 查看某张表的历史SQL1.查看MySQL通用查询日志(需提前开启)2.查看慢查询日志3.

MySQL底层文件的查看和修改方法

《MySQL底层文件的查看和修改方法》MySQL底层文件分为文本类(可安全查看/修改)和二进制类(禁止手动操作),以下按「查看方法、修改方法、风险管控三部分详细说明,所有操作均以Linux环境为例,需... 目录引言一、mysql 底层文件的查看方法1. 先定位核心文件路径(基础前提)2. 文本类文件(可直

MySQL数据目录迁移的完整过程

《MySQL数据目录迁移的完整过程》文章详细介绍了将MySQL数据目录迁移到新硬盘的整个过程,包括新硬盘挂载、创建新的数据目录、迁移数据(推荐使用两遍rsync方案)、修改MySQL配置文件和重启验证... 目录1,新硬盘挂载(如果有的话)2,创建新的 mysql 数据目录3,迁移 MySQL 数据(推荐两