3.5.4 聚合模型¶
聚合数据模型,也称为 Aggregate
数据模型。可根据 Key
列聚合数据, Doris
存储层保留聚合后的数据,从而可以减少存储空间和提升查询性能;通常用于需要汇总或聚合信息(如总数或平均值)的情况。
下面以实际的例子来说明什么是聚合模型,以及如何正确的使用聚合模型。
1 导入数据聚合¶
假设业务有如下数据表模式:
ColumnName | Type | AggregationType | Comment |
---|---|---|---|
user_id | LARGEINT | 用户 id | |
date | DATE | 数据灌入日期 | |
city | VARCHAR(20) | 用户所在城市 | |
age | SMALLINT | 用户年龄 | |
sex | TINYINT | 用户性别 | |
last_visit_date | DATETIME | REPLACE | 用户最后一次访问时间 |
cost | BIGINT | SUM | 用户总消费 |
max_dwell_time | INT | MAX | 用户最大停留时间 |
min_dwell_time | INT | MIN | 用户最小停留时间 |
如果转换成建表语句则如下(省略建表语句中的 Partition
和 Distribution
信息)
SQL | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
这是一个典型的用户信息和访问行为的事实表。在一般星型模型中,用户信息和访问行为一般分别存放在维度表和事实表中。这里为了更加方便的解释 Doris
的数据模型,将两部分信息统一存放在一张表中。
表中的列按照是否设置了 AggregationType
,分为 Key
(维度列)和 Value
(指标列)。没有设置 AggregationType
的 user_id
、 date
、 age
、 sex
称为 Key
,而设置了 AggregationType
的称为 Value
。
当导入数据时,对于 Key
列相同的行会聚合成一行,而 Value
列会按照设置的 AggregationType
进行聚合。 AggregationType
目前有以下几种聚合方式和 agg_state
:
-
SUM
:求和,多行的Value
进行累加。 -
REPLACE
:替代,下一批数据中的Value
会替换之前导入过的行中的Value
。 -
MAX
:保留最大值。 -
MIN
:保留最小值。 -
REPLACE_IF_NOT_NULL
:非空值替换。和REPLACE
的区别在于对于null
值,不做替换。 -
HLL_UNION
:HLL
类型的列的聚合方式,通过HyperLogLog
算法聚合。 -
BITMAP_UNION
:BIMTAP
类型的列的聚合方式,进行位图的并集聚合。
Warning
如果这几种聚合方式无法满足需求,则可以选择使用 agg_state
类型。
假设有以下导入数据(原始数据):
user_id | date | city | age | sex | last_visit_date | cost | max_dwell_time | min_dwell_time |
---|---|---|---|---|---|---|---|---|
10000 | 2017/10/1 | 北京 | 20 | 0 | 2017/10/1 6:00 | 20 | 10 | 10 |
10000 | 2017/10/1 | 北京 | 20 | 0 | 2017/10/1 7:00 | 15 | 2 | 2 |
10001 | 2017/10/1 | 北京 | 30 | 1 | 2017/10/1 17:05 | 2 | 22 | 22 |
10002 | 2017/10/2 | 上海 | 20 | 1 | 2017/10/2 12:59 | 200 | 5 | 5 |
10003 | 2017/10/2 | 广州 | 32 | 0 | 2017/10/2 11:20 | 30 | 11 | 11 |
10004 | 2017/10/1 | 深圳 | 35 | 0 | 2017/10/1 10:00 | 100 | 3 | 3 |
10004 | 2017/10/3 | 深圳 | 35 | 0 | 2017/10/3 10:20 | 11 | 6 | 6 |
通过 SQL
导入数据:
SQL | |
---|---|
1 2 3 4 5 6 7 8 |
|
这是一张记录用户访问某商品页面行为的表。以第一行数据为例,解释如下:
数据 | 说明 |
---|---|
10000 | 用户 id,每个用户唯一识别 id |
2017/10/1 | 数据入库时间,精确到日期 |
北京 | 用户所在城市 |
20 | 用户年龄 |
0 | 性别男(1 代表女性) |
2017/10/1 6:00 | 用户本次访问该页面的时间,精确到秒 |
20 | 用户本次访问产生的消费 |
10 | 用户本次访问,驻留该页面的时间 |
10 | 用户本次访问,驻留该页面的时间(冗余) |
那么当这批数据正确导入到 Doris
中后, Doris
中最终存储如下:
user_id | date | city | age | sex | last_visit_date | cost | max_dwell_time | min_dwell_time |
---|---|---|---|---|---|---|---|---|
10000 | 2017/10/1 | 北京 | 20 | 0 | 2017/10/1 7:00 | 35 | 10 | 2 |
10001 | 2017/10/1 | 北京 | 30 | 1 | 2017/10/1 17:05 | 2 | 22 | 22 |
10002 | 2017/10/2 | 上海 | 20 | 1 | 2017/10/2 12:59 | 200 | 5 | 5 |
10003 | 2017/10/2 | 广州 | 32 | 0 | 2017/10/2 11:20 | 30 | 11 | 11 |
10004 | 2017/10/1 | 深圳 | 35 | 0 | 2017/10/1 10:00 | 100 | 3 | 3 |
10004 | 2017/10/3 | 深圳 | 35 | 0 | 2017/10/3 10:20 | 11 | 6 | 6 |
可以看到,用户 10000
只剩下了一行聚合后的数据。而其余用户的数据和原始数据保持一致。对于用户 10000
聚合后的数据,前 5
列没有变化:
-
第
6
列值为2017-10-01 07:00:00
。因为last_visit_date
列的聚合方式为REPLACE
,所以2017-10-01 07:00:00
替换了2017-10-01 06:00:00
保存了下来。注意:在同一个导入批次中的数据,对于REPLACE
这种聚合方式,替换顺序不做保证,如在这个例子中,最终保存下来的,也有可能是2017-10-01 06:00:00
;而对于不同导入批次中的数据,可以保证,后一批次的数据会替换前一批次。 -
第
7
列值为35
:因为cost
列的聚合类型为SUM
,所以由20 + 15
累加获得35
。 -
第
8
列值为10
:因为max_dwell_time
列的聚合类型为MAX
,所以10
和2
取最大值,获得10
。 -
第
9
列值为2
:因为min_dwell_time
列的聚合类型为MIN
,所以10
和2
取最小值,获得2
。
经过聚合, Doris
中最终只会存储聚合后的数据。换句话说,即明细数据会丢失,用户不能够再查询到聚合前的明细数据了。
2 导入数据与已有数据聚合¶
假设现在表中已经拥有了前面导入的数据:
user_id | date | city | age | sex | last_visit_date | cost | max_dwell_time | min_dwell_time |
---|---|---|---|---|---|---|---|---|
10000 | 2017/10/1 | 北京 | 20 | 0 | 2017/10/1 7:00 | 35 | 10 | 2 |
10001 | 2017/10/1 | 北京 | 30 | 1 | 2017/10/1 17:05 | 2 | 22 | 22 |
10002 | 2017/10/2 | 上海 | 20 | 1 | 2017/10/2 12:59 | 200 | 5 | 5 |
10003 | 2017/10/2 | 广州 | 32 | 0 | 2017/10/2 11:20 | 30 | 11 | 11 |
10004 | 2017/10/1 | 深圳 | 35 | 0 | 2017/10/1 10:00 | 100 | 3 | 3 |
10004 | 2017/10/3 | 深圳 | 35 | 0 | 2017/10/3 10:20 | 11 | 6 | 6 |
再导入一批新的数据:
user_id | date | city | age | sex | last_visit_date | cost | max_dwell_time | min_dwell_time |
---|---|---|---|---|---|---|---|---|
10004 | 2017/10/3 | 深圳 | 35 | 0 | 2017/10/3 11:22 | 44 | 19 | 19 |
10005 | 2017/10/3 | 长沙 | 29 | 1 | 2017/10/3 18:11 | 3 | 1 | 1 |
通过 SQL
导入数据:
SQL | |
---|---|
1 2 3 |
|
那么当这批数据正确导入到 Doris
中后, Doris
中最终存储如下:
user_id | date | city | age | sex | last_visit_date | cost | max_dwell_time | min_dwell_time |
---|---|---|---|---|---|---|---|---|
10000 | 2017/10/1 | 北京 | 20 | 0 | 2017/10/1 7:00 | 35 | 10 | 2 |
10001 | 2017/10/1 | 北京 | 30 | 1 | 2017/10/1 17:05 | 2 | 22 | 22 |
10002 | 2017/10/2 | 上海 | 20 | 1 | 2017/10/2 12:59 | 200 | 5 | 5 |
10003 | 2017/10/2 | 广州 | 32 | 0 | 2017/10/2 11:20 | 30 | 11 | 11 |
10004 | 2017/10/1 | 深圳 | 35 | 0 | 2017/10/1 10:00 | 100 | 3 | 3 |
10004 | 2017/10/3 | 深圳 | 35 | 0 | 2017/10/3 11:22 | 55 | 19 | 6 |
10005 | 2017/10/3 | 长沙 | 29 | 1 | 2017/10/3 18:11 | 3 | 1 | 1 |
可以看到,用户 10004
的已有数据和新导入的数据发生了聚合。同时新增了 10005
用户的数据。
数据的聚合,在 Doris
中有如下三个阶段发生:
-
每一批次数据导入的
ETL
阶段。该阶段会在每一批次导入的数据内部进行聚合。 -
底层
BE
进行数据Compaction
的阶段。该阶段,BE
会对已导入的不同批次的数据进行进一步的聚合。 -
数据查询阶段。在数据查询时,对于查询涉及到的数据,会进行对应的聚合。
数据在不同时间,可能聚合的程度不一致。比如一批数据刚导入时,可能还未与之前已存在的数据进行聚合。但是对于用户而言,用户只能查询到聚合后的数据。即不同的聚合程度对于用户查询而言是透明的。用户需始终认为数据以最终的完成的聚合程度存在,而不应假设某些聚合还未发生。(可参阅聚合模型的局限性一节获得更多详情。)
3 agg_state¶
AGG_STATE
不能作为 key
列使用,建表时需要同时声明聚合函数的签名。
用户不需要指定长度和默认值。实际存储的数据大小与函数实现有关。
详情可见于 agg_state
。
建表
SQL | |
---|---|
1 2 3 4 5 6 7 8 9 |
|
其中 agg_state
用于声明数据类型为 agg_state
, sum/group_concat
为聚合函数的签名。注意 agg_state
是一种数据类型,同 int/array/string
agg_state
只能配合 state /merge/union
函数组合器使用。
agg_state
是聚合函数的中间结果,例如,聚合函数 sum
,则 agg_state
可以表示 sum(1,2,3,4,5)
的这个中间状态,而不是最终的结果。
agg_state
类型需要使用 state
函数来生成,对于当前的这个表,则为 sum_state
, group_concat_state
。
SQL | |
---|---|
1 2 3 |
|
此时表只有一行(注意,下面的表只是示意图,不是真的可以 select
显示出来)
k1 | k2 | k3 |
---|---|---|
1 | sum(1,2,3) | group_concat_state(a,b,c) |
再插入一条数据
SQL | |
---|---|
1 |
|
此时表的结构为
k1 | k2 | k3 |
---|---|---|
1 | sum(1,2,3) | group_concat_state(a,b,c) |
2 | sum(4) | group_concat_state(d) |
我们可以通过 merge
操作来合并多个 state
,并且返回最终聚合函数计算的结果
SQL | |
---|---|
1 2 3 4 5 6 |
|
sum_merge
会先把 sum(1,2,3)
和 sum(4)
合并成 sum(1,2,3,4)
,并返回计算的结果。因为 group_concat
对于顺序有要求,所以结果是不稳定的。
SQL | |
---|---|
1 2 3 4 5 6 |
|
如果不想要聚合的最终结果,可以使用 union
来合并多个聚合的中间结果,生成一个新的中间结果。
SQL | |
---|---|
1 |
|
此时的表结构为
k1 | k2 | k3 |
---|---|---|
1 | sum(1,2,3) | group_concat_state(a,b,c) |
2 | sum(4) | group_concat_state(d) |
3 | sum(1,2,3,4) | group_concat_state(a,b,c,d) |
可以通过查询
SQL | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
用户可以通过 agg_state
做出更细致的聚合函数操作。
注意 agg_state
存在一定的性能开销。