网站首页 > 技术文章 正文
一,概述
一般而言,我们对关系型数据库系统,进行表结构设计时,会按数据的种类,进行分类,一般有如下种类:
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;另在某个时间字段上,加上索引。比如:
CREATE TABLE ow_pkg.TT_FLOW_IN
(
IN_UUID varchar2(32),
IN_SHEET_CD varchar2(255) NOT NULL,
IN_TIME date NOT NULL,
SEND_NODE_ID decimal(38,0) NOT NULL,
RECEIVE_NODE_ID decimal(38,0) NOT NULL,
CREATED_BY varchar2(20),
CREATED_DT date,
UPDATED_BY varchar2(20),
UPDATED_DT date,
UPDATE_CNT INTEGER DEFAULT 0 NOT NULL
)
;
其中, 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),正常情况下,数据的存储也是按时间顺序的。
对于业务交易明细表,一般创建明细表主键、在明细表指向主表的字段上创建普通索引。比如:
CREATE TABLE ow_pkg.TT_FLOW_IN_DETAIL
(
IN_DETAIL_UUID varchar2(32), --pk
IN_UUID varchar2(32), --fk
PROJ_ID decimal(38,0) NOT NULL,
STATUS_ID decimal(38,0),
CONTAINER_ID decimal(38,0) NOT NULL,
REAL_QTY decimal(10,0),
PLAN_QTY decimal(10,0),
CREATED_BY varchar2(20),
CREATED_DT date,
UPDATED_BY varchar2(20),
UPDATED_DT date,
UPDATE_CNT INTEGER DEFAULT 0 NOT NULL,
)
;
CREATE UNIQUE INDEX idx_tt_flow_in_detail_in_detail_uuid ON ow_pkg.TT_FLOW_IN_DETAIL(IN_DETAIL_UUID);
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):
SELECT * FROM (
SELECT ROW_NUMBER() OVER (ORDER BY i.in_time desc,i.IN_SHEET_CD,i.in_uuid ) as rownum_xx
,i.*
from TT_FLOW_IN i
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')
and i.IN_SHEET_CD is not null
)
WHERE rownum_xx >= 0 and rownum_xx <= 20;
请主要,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 秒。
性能可以说是非常的好。
猜你喜欢
- 2025-07-17 PageHelper - 最方便的 MyBatis 分页插件
- 2025-07-17 面试二:pagehelper是怎么实现分页的,
- 2025-07-17 MyBatis如何实现分页查询?(mybatis-plus分页查询)
- 2025-07-17 SpringBoot 各种分页查询方式详解(全网最全)
- 2024-10-24 SpringBoot源码:pageHelper分页,值得你看
- 2024-10-24 mysql和oracle的区别有哪些 mysql和oracle关系
- 2024-10-24 java必背综合知识点总结(数据库篇)
- 2024-10-24 如果让你写一个MyBatis分页插件,你打算怎么实现?
- 2024-10-24 PageHelper分页插件源码及原理剖析
- 2024-10-24 mysql和oracle的区别有什么 oracle 和mysql的区别
你 发表评论:
欢迎- 613℃几个Oracle空值处理函数 oracle处理null值的函数
- 605℃Oracle分析函数之Lag和Lead()使用
- 593℃0497-如何将Kerberos的CDH6.1从Oracle JDK 1.8迁移至OpenJDK 1.8
- 591℃Oracle数据库的单、多行函数 oracle执行多个sql语句
- 584℃Oracle 12c PDB迁移(一) oracle迁移到oceanbase
- 579℃【数据统计分析】详解Oracle分组函数之CUBE
- 568℃最佳实践 | 提效 47 倍,制造业生产 Oracle 迁移替换
- 559℃Oracle有哪些常见的函数? oracle中常用的函数
- 最近发表
-
- PageHelper - 最方便的 MyBatis 分页插件
- 面试二:pagehelper是怎么实现分页的,
- MyBatis如何实现分页查询?(mybatis-plus分页查询)
- SpringBoot 各种分页查询方式详解(全网最全)
- 如何在Linux上运行exe文件,怎么用linux运行windows软件
- 快速了解hive(快速了解美国50个州)
- Python 中的 pyodbc 库(pydbclib)
- Linux搭建Weblogic集群(linux weblogic部署项目步骤)
- 「DM专栏」DMDSC共享集群之部署(一)——共享存储配置
- 故障分析 | MySQL 派生表优化(mysql pipe)
- 标签列表
-
- 前端设计模式 (75)
- 前端性能优化 (51)
- 前端模板 (66)
- 前端跨域 (52)
- 前端缓存 (63)
- 前端aes加密 (58)
- 前端脚手架 (56)
- 前端md5加密 (54)
- 前端路由 (61)
- 前端数组 (73)
- 前端js面试题 (50)
- 前端定时器 (59)
- 前端获取当前时间 (50)
- Oracle RAC (76)
- oracle恢复 (77)
- oracle 删除表 (52)
- oracle 用户名 (80)
- oracle 工具 (55)
- oracle 内存 (55)
- oracle 导出表 (62)
- oracle约束 (54)
- oracle 中文 (51)
- oracle链接 (54)
- oracle的函数 (58)
- 前端调试 (52)
本文暂时没有评论,来添加一个吧(●'◡'●)