单个表上亿行数据的主键、索引设计,及分页查询「终于解决」

单个表上亿行数据的主键、索引设计,及分页查询「终于解决」单个表数据量超过1亿的,需要精心设计表的主键、索引,其分页查询也不能乱写,否则性能不佳。
此文章特介绍作者心得。

单个表上亿行数据的主键、索引设计,及分页查询

一,概述

一般而言,我们对关系型数据库系统,进行表结构设计时,会按数据的种类,进行分类,一般有如下种类:

1)主数据,其数据量基本稳定,不随时间而线性增长。比如,分公司,产品,经销商。 这种数据库表,我们一般以 tm_ 作为表名的前缀, 意思是 table of master data。
2)系统级数据,其数据量基本稳定,不随时间而线性增长。比如,用户权限控制,配置参数。 这种数据库表,我们一般以 ts_ 作为表名的前缀, 意思是 table of system。
3)日志数据表,随时间而线性增长,但会安排定时任务定期删除旧数据,保持总体数据量稳定。 这种数据库表,我们一般以 tl_ 作为表名的前缀, 意思是 table of log。
4)接口数据表,随时间而线性增长,但会安排定时任务定期删除旧数据,保持总体数据量稳定。 这种数据库表,我们一般以 ti_ 作为表名的前缀, 意思是 table of interface。
5)业务交易数据,随时间而线性增长,用户平常关注最近若干天的数据,少数情况下会查阅很久以前的数据。 这种数据库表,我们一般以 tt_ 作为表名的前缀, 意思是 table of transaction data。
6)关系数据,可能是以上 1,2,5 的关系表,我们分别以 tmr_, tsr_, ttr_ 作为表名的前缀。

通常,数据量大的,都是上述“5. 业务交易数据”

 

二、业务交易表的主键、索引设计

业务交易数据,按通常的理解,一般有主表、明细表两种。

业务交易主表的主键,一般是 id/uuid;另在某个时间字段上,加上索引。比如:

 1 CREATE TABLE ow_pkg.TT_FLOW_IN
 2 (
 3    IN_UUID varchar2(32),
 4    IN_SHEET_CD varchar2(255) NOT NULL,
 5    IN_TIME date NOT NULL,
 6 
 7    SEND_NODE_ID decimal(38,0) NOT NULL,
 8    RECEIVE_NODE_ID decimal(38,0) NOT NULL,
 9 
10    CREATED_BY varchar2(20),
11    CREATED_DT date,
12    UPDATED_BY varchar2(20),
13    UPDATED_DT date,
14    UPDATE_CNT INTEGER DEFAULT 0  NOT NULL
15 )
16 ;

其中, in_uuid 为主键。

对于交易主表的主键,可用按 SQL 语法,创建 primary key, 也可以只创建成唯一索引(UNIQUE INDEX), 或普通索引(INDEX)。之所以会有这种的做法,是因为有的数据库,比如 MS SQL Server, 默认在主键上创建聚集索引(clustered index, 不同的数据库,名词可能有所差异),数据的存储,按主键的数值顺序,如果我们使用 uuid 做主键,这可能不是我们期望的。
在主键上创建普通索引,是在使用 uuid 作为主键数据时。因 uuid 本身就能保证数据的唯一性,不需要使用数据库的 primary key 或 UNIQUE INDEX 语法来保证数据唯一性。且有的架构师,担心每行数据 insert 到表时,拥有 primary key 或 UNIQUE INDEX 定义的表,数据库会自动进行主键数据的唯一性检查,如果数据量极大,这个唯一性检查的步骤有可能需要花费额外的时间,还不如使用普通索引,跳过主键数据的唯一性检查。

这里我们创建唯一性索引。

CREATE UNIQUE INDEX idx_tt_flow_in_in_uuid ON ow_pkg.TT_FLOW_IN(IN_UUID); 

一般在交易主表的某个时间字段上,创建普通索引,或者聚集索引(clustered index),比如:

CREATE INDEX idx_tt_flow_in_in_time ON ow_pkg.TT_FLOW_IN(IN_TIME);

交易表的数据,一般是 insert 多、delete 少,如果不定义主键、不创建聚集索引(clustered index),正常情况下,数据的存储也是按时间顺序的。

对于业务交易明细表,一般创建明细表主键、在明细表指向主表的字段上创建普通索引。比如:

 1 CREATE TABLE ow_pkg.TT_FLOW_IN_DETAIL
 2 (
 3    IN_DETAIL_UUID varchar2(32),                --pk
 4    IN_UUID varchar2(32),                    --fk
 5    PROJ_ID decimal(38,0) NOT NULL,
 6    STATUS_ID decimal(38,0),
 7    CONTAINER_ID decimal(38,0) NOT NULL,
 8    REAL_QTY decimal(10,0),
 9    PLAN_QTY decimal(10,0),
10    CREATED_BY varchar2(20),
11    CREATED_DT date,
12    UPDATED_BY varchar2(20),
13    UPDATED_DT date,
14    UPDATE_CNT INTEGER DEFAULT 0  NOT NULL,
15 )
16 ;
17 CREATE UNIQUE INDEX idx_tt_flow_in_detail_in_detail_uuid ON ow_pkg.TT_FLOW_IN_DETAIL(IN_DETAIL_UUID);
18 CREATE INDEX idx_tt_flow_in_detail_in_uuid ON ow_pkg.TT_FLOW_IN_DETAIL(IN_UUID);

交易明细表不需要在某个时间字段上,创建索引。此时基于 in_uuid 查找 tt_flow_in_detail 表,数据量不会超过 30 行。

 

三、分页查询

SQL 标准中,有分页查询的语法。一般只针对业务主表进行查询分页、然后点击查找结果的某行,弹出窗口显示业务明细表数据。

这里的分页查询 SQL 为(基于 Oracle):

1 SELECT * FROM (
2     SELECT ROW_NUMBER() OVER (ORDER BY i.in_time desc,i.IN_SHEET_CD,i.in_uuid ) as rownum_xx
3     ,i.*
4     from TT_FLOW_IN i
5     where i.in_time between to_date("2020-01-01 00:00" ,"yyyy-mm-dd hh24:mi") and to_date("2020-01-02 00:00" ,"yyyy-mm-dd hh24:mi")
6     and i.IN_SHEET_CD is not null
7 )
8 WHERE rownum_xx >= 0 and rownum_xx <= 20;

以上 SQL 的 where 中的参数,可以动态参数。比如对于 java ,可以使用占位符 ? ,使用 Java 的 PreparedStatement , 进行执行。

通常大家忽略的是 order by 这部分。这一部分一般按顺序依次为: 业务主表的时间字段(逆序排序)、业务主表的单证编号、其它可见字段、业务主表的主键

不加排序(order by) 的分页是耍流氓,没意义的;排序字段中必须包含用户能理解的数据项,如果只按后台 id/uuid 排序,用户会觉得数据混乱无序;如果 order by 最后不加主键,有可能导致某些行的数据,既出现在第 n 页、又出现在第 n+1 页。

 

四、分页查询的性能

以上分页查询 SQL, 在单个表数据量为 1.3 亿行的情况下,查询时间范围跨度为 15 天的情况下,每查询一次改一下查询时间范围的小时数,多次测试,分别用时:

0.047 秒、0.062 秒、0.047 秒、0.062 秒。

平均用时 0.055 秒

性能可以说是非常的好。

=======欢迎转载,转载请注明出处,https://www.cnblogs.com/jacklondon/
 

转载请注明出处: http://www.cnblogs.com/jacklondon ; 欢迎访问 http://www.zheguisoft.com/ 并提建议。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/5885.html

(0)
上一篇 2023-04-21
下一篇 2023-04-21

相关推荐

  • 快速理解Python Sort排序算法原理与应用

    快速理解Python Sort排序算法原理与应用排序算法是计算机程序设计中最常见的问题之一。排序算法将一组数据按照一定的规则进行排列,使得数据之间具有一定的有序性。根据排序算法的处理方式不同,同样的数据集合可能会有不同的排列结果。

    2024-02-07
    99
  • windows安装oracle11g_window10数据库安装

    windows安装oracle11g_window10数据库安装一、下载下载oracle11g安装文件,一般包括win64_11gR2_database_1of2和win64_11gR2_database_2of2 2个文件,下载完成后解压到一个文件中: 下载文件

    2023-02-11
    154
  • MySQL中distinct的使用方法【转】[通俗易懂]

    MySQL中distinct的使用方法【转】[通俗易懂]一、基本使用 distinct一般是用来去除查询结果中的重复记录的,而且这个语句在 、`insert delete update`中只可以在select中使用,具体的语法如下: 这里的expressi

    2023-02-17
    150
  • mongodb设计规范_mongodb菜鸟教程

    mongodb设计规范_mongodb菜鸟教程最近刚入坑MongoDB,感觉比MySQL扩展性更强,一张表可以存储特别复杂的字段,这点我非常喜欢,最近需要用MongoDB存储一篇文章的数据,文章的评论和回复的数据存储是个大问题,设计了好久感觉我设

    2023-04-15
    181
  • setup airflow on MySQL[亲测有效]

    setup airflow on MySQL[亲测有效]
    SQLite Database https://airflow.apache.org/docs/apache-airflow/stable/howto/se…

    2023-04-18
    170
  • 今日头条在消息服务平台和容灾体系建设方面的实践与思考

    今日头条在消息服务平台和容灾体系建设方面的实践与思考本篇文章整理自今日头条的沈辉在 RocketMQ 开发者沙龙中的演讲,主要和大家分享一下,RocketMQ 在微服务架构下的实践和容灾体系建设。沈辉是今日头条的架构师,主要负责 RocketMQ 在…

    2022-12-18
    148
  • 使用Python字符串扩充标题

    使用Python字符串扩充标题在许多编程应用中,标题是必不可少的一个组成部分,无论是在GUI程序的窗口标题栏中,还是在文档或网站的主题中,其作用都是十分重要的。然而,在有些情况下,我们需要增加标题的变化和特效来提高页面的吸引力和用户友好性。比如,当我们希望将标题的字体变得更大或更小,或者需要在标题的两边添加特殊字符,这时使用Python字符串来扩充标题就变得尤为重要。

    2024-09-16
    22
  • PyCharm终端位置

    PyCharm终端位置PyCharm是一款功能强大的Python IDE,它提供了很多方便开发的工具和功能。在PyCharm中,终端也是一个非常重要的工具,它可以方便地执行命令,例如调试Python程序、运行pip命令等。

    2024-07-21
    38

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注