百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

如何利用数仓创建时序表

wptr33 2025-02-07 19:21 27 浏览

摘要:怎么快速上手,创建适合自己业务的时序表,怎样使用才能真正发挥时序表的优势。

本文分享自华为云社区《GaussDB(DWS)带你走近IoT时代-时序表建表最佳实践-云社区-华为云》,作者:AndyCao。

创建第一张时序表

我们首先来介绍一下如何创建一张时序表

语法格式


CREATE TABLE [ IF NOT EXISTS ] table_name

({ column_name data_type [ kv_type ]

| LIKE source_table [like_option [...] ] }

}

[, ... ])

[ WITH ( {storage_parameter = value} [, ... ] ) ]

[ TABLESPACE tablespace_name ]

[ DISTRIBUTE BY HASH ( column_name [,...])]

[ TO { GROUP groupname | NODE ( nodename [, ... ] ) } ]

[ table_partitioning_clauses ]

[ PARTITION BY {

{RANGE (partition_key) ( partition_less_than_item [, ... ] )}

} [ { ENABLE | DISABLE } ROW MOVEMENT ] ];

其中like选项like_option为:{ INCLUDING | EXCLUDING } { DEFAULTS | CONSTRAINTS | INDEXES | STORAGE | COMMENTS | PARTITION | RELOPTIONS | DISTRIBUTION | ALL }

时序表的建表语法,在很大程度上继承了行存和列存的语法,降低了用户的学习成本,能够更容易理解和使用。我们上篇博客中介绍到,将时序表的列分为三种kv_type类型(tag、field、time),那么如何将对应的列指定为合适的类型,帮助我们更好的提高导入、查询等场景的性能,让业务场景运行的更加高效呢?

我们继续以发电机组的场景作为示例,创建一张存储发电机组采样数据的时序表:


CREATE TABLE IF NOT EXISTS GENERATOR(

dynamo text TSTag,

manufacturer text TSTag,

model text TSTag,

location text TSTag,

ID bigint TSTag,

voltage numeric TSField,

power bigint TSTag,

frequency numeric TSField,

angle numeric TSField,

time timestamptz TSTime) with (orientation=TIMESERIES, period='7 days', ttl='1 month') distribute by hash(model);

我们可以看到,上述建表语句中:

  • 对于不随时间的变化而变化,描述发电机的属性信息的列(发电机信息、生产厂商、型号、位置、ID)被设置为tag列,在建表时需要将对应的列后面指定为TSTag;
  • 对于采样数据的维度(电压、功率、频率、电流相角)这些对应的采样数值随时间的变化而变,我们将这些维度设置为field列,建表语句数据类型后面指定为TSField;
  • 最后一列我们指定为时间列time,存储field列数据对应的时间信息,建表时将指定为TSTime。

在写建表语句时,对于tag列的顺序,我们可以适当优化一下,将唯一性(distinct值)较高的列尽量写在前面,这样对于时序场景的性能有一些提升。如果用户没有手动优化,GaussDB(DWS) IoT数仓也可以自适应的帮助用户提高时序场景的性能,这后面我们会专门文章介绍这一黑科技

另外,创建时序表时一定要指定表级参数orientation属性设置为timeseries。时序表不需要手动指定DISTRIBUTE BY和PARTITION BY, 默认按照所有tag列分布,且分区健默认为tstim指定的时间列。

对于create table like语法,该语法需要自动从源表中继承列名和对应的kv_type类型。因此如果源表是非时序表,新表是时序表,对应的列的kv_type类型无法确定,则无法创建成功。

时序表列类型

上面对时序表的三种kv_type属性进行了简单的说明:(维度属性(TSTag),指标属性(TSField),时间属性(TSTime),那么每个列他们支持的类型(类似建表语句中的text、int、numeric等)分别都是什么呢?是否可以不设置kv_type类型?

首先时序表必须指定一个时间属性(TSTime),且只能指定一个,且TSTime类型的列不能被删除。至少存在一个TSTagTSField列,否则建表报错。

TSTag列支持类型:text, char, bool, int, big int。

TSTime列支持类型:timestamp with time zone, timestamp without time zone。在兼容Oracle语法的数据库中,也支持date类型。涉及到时区相关操作时,请选择带时区的时间类型。

TSField列支持的数据类型同列存表保持一致

自动设置分区边界

时序表具备数据生命周期管理的能力。每天数以亿计的数据不间断涌入,对于很久之前的数据,其价值较低不经常访问,可以定期将无用的老数据删除。并且由于最新的数据不断进来,需要定期给表增加新的分区,避免新数据无法存储。因此时序表需要具备定时增加分区和定时删除分区的能力。

时序表以TSTIME列为分区键,创建具有自动分区管理功能的分区表,帮助我们大大减少运维操作的工作。在上面的建表语句中,在表级参数项中可以看到,时序表指定了自动分区管理两个参数period和ttl。

  • period:设置自动创建分区的间隔时间,默认值为1 day, 取值范围:1 hour ~ 100 years。默认会为时序表创建自增分区任务。自增分区任务动态为我们创建分区,保证当前时刻有足够充裕的分区用于导入数据。
  • ttl:设置自动淘汰分区的时间,取值范围:1 hour ~ 100 years。默认不创建淘汰分区任务,需要用户自己在建表手动指定,或者建表后通过ALTER TABLE语法设置。淘汰分区的策略是通过计算 nowtime - 分区boundary > ttl,满足该条件的分区将被drop掉。帮助用户定时清理过期的旧数据。

上面举得例子中,时序表建表语句没有指定分区,那么分区的起始时间是怎么样的。为了方便用户的使用,我们的分区边界的设置分为了一下几种情况

  • period设置为“小时” , 分区起始边界值为下个小时整点,分区的间隔为period的值
  • period设置为“天” , 分区起始边界值为第二天零点,分区的间隔为period的值
  • period设置为“月” , 分区起始边界值为下个月零点,分区的间隔为period的值
  • period设置为“年” , 分区起始边界值为明年零点,分区的间隔为period的值

tsdb=# CREATE TABLE IF NOT EXISTS GENERATOR(

tsdb(# dynamo text TSTag,

tsdb(# manufacturer text TSTag,

tsdb(# model text TSTag,

tsdb(# location text TSTag,

tsdb(# ID bigint TSTag,

tsdb(# voltage numeric TSField,

tsdb(# power bigint TSTag,

tsdb(# frequency numeric TSField,

tsdb(# angle numeric TSField,

tsdb(# time timestamptz TSTime) with (orientation=TIMESERIES, period='1 hour', ttl='1 month') distribute by hash(model);

CREATE TABLE

tsdb=# select now();

now

-------------------------------

2022-05-25 15:28:38.520757+08

(1 row)

tsdb=# select relname, boundaries from pg_partition where parentid=(select oid from pg_class where relname='generator') order by boundaries ;

relname | boundaries

----------------+----------------------------

default_part_1 | {"2022-05-25 16:00:00+08"}

default_part_2 | {"2022-05-25 17:00:00+08"}

p1653505200 | {"2022-05-26 03:00:00+08"}

p1653541200 | {"2022-05-26 13:00:00+08"}

p1653577200 | {"2022-05-26 23:00:00+08"}

......

这样的使用方式,能够使得用户能够便捷快速的创建时序表。当然我们也支持在建表时用户手动指定分区边界的起始值。关于自动分区管理功能更详细的描述,可以参考《GaussDB(DWS) 分区自动管理介绍》。


tsdb=# select now();

now

-------------------------------

2022-05-31 20:36:09.700096+08

(1 row)

tsdb=# CREATE TABLE IF NOT EXISTS GENERATOR(

tsdb(# dynamo text TSTag,

tsdb(# manufacturer text TSTag,

tsdb(# model text TSTag,

tsdb(# location text TSTag,

tsdb(# ID bigint TSTag,

tsdb(# voltage numeric TSField,

tsdb(# power bigint TSTag,

tsdb(# frequency numeric TSField,

tsdb(# angle numeric TSField,

tsdb(# time timestamptz TSTime) with (orientation=TIMESERIES, period='1 day') distribute by hash(model)

tsdb-# partition by range(time)

tsdb-# (

tsdb(# PARTITION P1 VALUES LESS THAN('2022-05-30 16:32:45'),

tsdb(# PARTITION P2 VALUES LESS THAN('2022-05-31 16:56:12')

tsdb(# );

WARNING: partition boundary is less than current time.

CREATE TABLE

tsdb=# select relname, boundaries from pg_partition where parentid=(select oid from pg_class where relname='generator') order by boundaries ;

relname | boundaries

-------------+----------------------------

p1 | {"2022-05-30 16:32:45+08"}

p2 | {"2022-05-31 16:56:12+08"}

p1654073772 | {"2022-06-01 16:56:12+08"}

p1654160172 | {"2022-06-02 16:56:12+08"}

......

至此,我们已经建成了自己的第一张时序表,接下来我们会针对时序表的DDL各种操作进行详细的实践,帮助大家学好、用好时序数据库,更好的为用户的业务服务。

点击下方,第一时间了解华为云新鲜技术~

华为云博客_大数据博客_AI博客_云计算博客_开发者中心-华为云

相关推荐

高性能并发队列Disruptor使用详解

基本概念Disruptor是一个高性能的异步处理框架,是一个轻量的Java消息服务JMS,能够在无锁的情况下实现队列的并发操作Disruptor使用环形数组实现了类似队列的功能,并且是一个有界队列....

Disruptor一个高性能队列_java高性能队列

Disruptor一个高性能队列前言说到队列比较熟悉的可能是ArrayBlockingQueue、LinkedBlockingQueue这两个有界队列,大多应用在线程池中使用能保证线程安全,但其安全性...

谈谈防御性编程_防御性策略

防御性编程对于程序员来说是一种良好的代码习惯,是为了保护自己的程序在不可未知的异常下,避免带来更大的破坏性崩溃,使得程序在错误发生时,依然能够云淡风轻的处理,但很多程序员入行很多年,写出的代码依然都是...

有人敲门,开水开了,电话响了,孩子哭了,你先顾谁?

前言哎呀,这种情况你肯定遇到过吧!正在家里忙活着,突然——咚咚咚有人敲门,咕噜咕噜开水开了,铃铃铃电话响了,哇哇哇孩子又哭了...我去,四件事一起来,人都懵了!你说先搞哪个?其实这跟我们写Java多线...

面试官:线程池如何按照core、max、queue的执行顺序去执行?

前言这是一个真实的面试题。前几天一个朋友在群里分享了他刚刚面试候选者时问的问题:"线程池如何按照core、max、queue的执行循序去执行?"。我们都知道线程池中代码执行顺序是:co...

深入剖析 Java 中线程池的多种实现方式

在当今高度并发的互联网软件开发领域,高效地管理和利用线程资源是提升程序性能的关键。Java作为一种广泛应用于后端开发的编程语言,为我们提供了丰富的线程池实现方式。今天,就让我们深入探讨Java中...

并发编程之《彻底搞懂Java线程》_java多线程并发解决方案详解

目录引言一、核心概念:线程是什么?...

Redis怎么实现延时消息_redis实现延时任务

一句话总结Redis可通过有序集合(ZSET)实现延时消息:将消息作为value,到期时间戳作为score存入ZSET。消费者轮询用ZRANGEBYSCORE获取到期消息,配合Lua脚本保证原子性获取...

CompletableFuture真的用对了吗?盘点它最容易被误用的5个场景

在Java并发编程中,CompletableFuture是处理异步任务的利器,但不少开发者在使用时踩过这些坑——线上服务突然雪崩、异常悄无声息消失、接口响应时间翻倍……本文结合真实案例,拆解5个最容易...

接口性能优化技巧,有点硬_接口性能瓶颈

背景我负责的系统到2021年初完成了功能上的建设,开始进入到推广阶段。随着推广的逐步深入,收到了很多好评的同时也收到了很多对性能的吐槽。刚刚收到吐槽的时候,我们的心情是这样的:...

禁止使用这5个Java类,每一个背后都有一段"血泪史"

某电商平台的支付系统突然报警:大量订单状态异常。排查日志发现,同一笔订单被重复支付了三次。事后复盘显示,罪魁祸首竟是一行看似无害的SimpleDateFormat代码。在Java开发中,这类因使用不安...

无锁队列Disruptor原理解析_无锁队列实现原理

队列比较队列...

Java并发队列与容器_java 并发队列

【前言:无论是大数据从业人员还是Java从业人员,掌握Java高并发和多线程是必备技能之一。本文主要阐述Java并发包下的阻塞队列和并发容器,其实研读过大数据相关技术如Spark、Storm等源码的,...

线程池工具及拒绝策略的使用_线程池处理策略

线程池的拒绝策略若线程池中的核心线程数被用完且阻塞队列已排满,则此时线程池的资源已耗尽,线程池将没有足够的线程资源执行新的任务。为了保证操作系统的安全,线程池将通过拒绝策略处理新添加的线程任务。...

【面试题精讲】ArrayBlockingQueue 和 LinkedBlockingQueue 区别?

有的时候博客内容会有变动,首发博客是最新的,其他博客地址可能会未同步,认准...