0
点赞
收藏
分享

微信扫一扫

《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》

Item_sum类用于SQL聚合函数的特殊表达式基类。

这些表达式是在聚合函数(sum、max)等帮助下形成的。item_sum类也是window函数的基类。
聚合函数(Aggregate Function)实现的大部分代码在item_sum.h和item_sum.cc

聚合函数限制

不能在表达式的所有位置使用聚合函数,使用聚合函数应该有一些明确的限制。

在没有嵌套的查询中,select列表的表达式中,having自居中使用聚合函数是有效的,在where子句、form子句或group by子句则是无效的。

关于解释,这篇文章可以看看:​​javascript:void(0)​​

在带有嵌套子查询的查询中检测聚合函数是否有效的规则比较复杂:

如下列查询:

SELECT t1.a FROM t1 GROUP BY t1.a HAVING t1.a > ALL (SELECT t2.c FROM t2 WHERE SUM(t1.b) < t2.c).

在子查询的where子句中使用了聚合函数sum(),但是由于它包含在外部查询的having子句中,因此它是有效的。将针对主查询中定义的每个组而不是子查询的组来评估表达式sum。

又如下列查询:

SELECT t1.a FROM t1 GROUP BY t1.a HAVING t1.a > ALL(SELECT t2.c FROM t2 GROUP BY t2.c HAVING SUM(t1.a) < t2.c)

聚合函数可以在外部查询块和内部查询块中进行评估,如果我们为外部查询评估sum,则将得到t1.a乘上t1组中的基数。在这种情况下,sum(t1.a)被用作每个相关子查询中的常数。但是,也可以为内部查询评估sum。此时t1.a将是每个相关子查询的常数,并且对表t2的每个组执行求和。

因此,根据向哪个查询快分配聚合函数,可以获得不同的结果。

检测聚合函数的查询块的一般规则如下:

考虑一个聚合函数S(E),其中E是一个包含列引用C1,…,Cn的表达式。针对包含聚合函数S(E)的查询块Qi会解析所有列引用Ci。令Q为所有查询块Qi中最内部的查询块。(注意,S(E)绝对不能在包含子查询Q的查询的查询块聚合,否则S(E)将引用至少一个未绑定的列引用)。如果在允许聚合函数的Q的构造中使用函数S(E),则我们在Q中聚合S(E)。

否则:如果启用了ANSI SQL模式,则报告错误。如果没有启用,在允许使用S(E)的地方查找包含S(E)的最内部的查询块。聚合的位置取决于子查询包含在哪个子句中。当包含在where子句中,包含在选择列表中或包含在having子句时,结果不同。

一些成员说明

成员​​base_select​​包含对其中包含了聚合函数的查询块的引用。

成员​​aggr_select​​包含对其中使用了聚合函数的查询块的引用。

​max_aggr_level​​​字段保留聚合函数中包含的未绑定列引用的最大嵌套级别。在嵌套级别较小的子查询中不能包含聚合函数比​​max_aggr_level​​​高。可以聚合在嵌套级别大于​​max_aggr_level​​的子查询中。

如果聚合函数中不包含任何列引用,如count(*),则​​max_aggr_level​​为-1。

字段​​max_sum_func_level​​​将包含用作给定聚合函数的参数的子表达式的聚合函数的嵌套级别的最大值,但不会在此聚合函数内的任何子查询块中聚合。仅当​​s1.max_sum_func_level < s0.max_sum_func_level​​时,嵌套的聚合函数s1才能在聚合函数s0中使用。如果未在s0内的任何子查询中计算s1,则将聚合函数s1视为嵌套于集合函数s0。

当我们使用递归方法​​fix_fields​​​遍历查询子表达式时,将检查使用集合函数的条件。当我们将此方法用于Item_sum类的对象时,首先,在下降时(下降不知道指什么意思),调用​​init_sum_func_check​​​方法,该方法初始化检查时使用的成员。然后在上升过程中,调用方法​​check_sum_func​​​,该方法验证设置的函数的用法,并在无效时报告错误。方法​​check_sum_func​​​用于链接在包含的查询块中聚合的聚合函数的item。此类函数的循环链通过字段​​inner_sum_func_list​​​附加到相应的​​select_lex​​结构。

窗口函数

关于什么是窗口函数,参考链接​​通俗易懂的学会:SQL窗口函数​​

大部分的聚合函数如(sum、count、avg)也可以用作窗口函数。此时有如下限制:

1、不使用任何聚合器

2、不支持distinct

3、val_* ( ) 的作用不只是返回函数的当前值:它首先将函数的参数累加到函数的状态。例如处理​​end_write_wf()​​包含WF输入的临时表。每个输入行都传递给copy_funcs(),后者调用WF的 val_ *()对其进行累加。

类型判断

很多时候我们要判断聚合函数的类型,那么首先要先判断当前的操作是否是聚合操作,然后再判断聚合操作的类型。

Item类有一个纯虚函数: ​​virtual enum Type type() const =0;​​ 其对应的Type在该类中定义为:

enum Type {
INVALID_ITEM = 0,
FIELD_ITEM,
FUNC_ITEM,
SUM_FUNC_ITEM,
STRING_ITEM,
INT_ITEM,
REAL_ITEM,
NULL_ITEM,
VARBIN_ITEM,
COPY_STR_ITEM,
FIELD_AVG_ITEM,
DEFAULT_VALUE_ITEM,
PROC_ITEM,
COND_ITEM,
REF_ITEM,
FIELD_STD_ITEM,
FIELD_VARIANCE_ITEM,
INSERT_VALUE_ITEM,
SUBSELECT_ITEM,
ROW_ITEM,
CACHE_ITEM,
TYPE_HOLDER,
PARAM_ITEM,
TRIGGER_FIELD_ITEM,
DECIMAL_ITEM,
XPATH_NODESET,
XPATH_NODESET_CMP,
VIEW_FIXER_ITEM,
FIELD_BIT_ITEM,
VALUES_COLUMN_ITEM
};

Item_sum从Item派生出来,除了type函数外,他还带有一个函数​​virtual enum Sumfunctype sum_func () const=0;​​在Item_sum中定义了Sumfunctype的类型。

enum Sumfunctype {
COUNT_FUNC, // COUNT
COUNT_DISTINCT_FUNC, // COUNT (DISTINCT)
SUM_FUNC, // SUM
SUM_DISTINCT_FUNC, // SUM (DISTINCT)
AVG_FUNC, // AVG
AVG_DISTINCT_FUNC, // AVG (DISTINCT)
MIN_FUNC, // MIN
MAX_FUNC, // MAX
STD_FUNC, // STD/STDDEV/STDDEV_POP
VARIANCE_FUNC, // VARIANCE/VAR_POP and VAR_SAMP
SUM_BIT_FUNC, // BIT_AND, BIT_OR and BIT_XOR
UDF_SUM_FUNC, // user defined functions
GROUP_CONCAT_FUNC, // GROUP_CONCAT
JSON_AGG_FUNC, // JSON_ARRAYAGG and JSON_OBJECTAGG
ROW_NUMBER_FUNC, // Window functions
RANK_FUNC,
DENSE_RANK_FUNC,
CUME_DIST_FUNC,
PERCENT_RANK_FUNC,
NTILE_FUNC,
LEAD_LAG_FUNC,
FIRST_LAST_VALUE_FUNC,
NTH_VALUE_FUNC,
ROLLUP_SUM_SWITCHER_FUNC
};

可以通过这些类型来判断读当前具体的聚合操作。除此之外,还可以利用lex的一些helper函数来进行辅助判断,如​​is_single_grouped​

bool is_single_grouped() const {
return m_agg_func_used && group_list.elements == 0 &&
m_having_cond == nullptr;
}

如果该函数返回true,则说明没有group by 也没有have

主要的聚合函数具体在代码中的类结构和继承关系

《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_聚合函数
COUNT/SUM/AVG/STD/VAR_POP函数:
《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_数据库_02
MIN/MAX函数:
《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_sql_03
BIT_OR/BIT_AND/BIT_XOR函数:
《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_子查询_04

聚合过程(不带group by)

不带group by的聚合会使用辅助类​​Aggregator​​​,而group by并不使用该辅助类。
《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_聚合函数_05

在优化阶段需要进行setup,比如初始化distinct或者sorting需要临时表或者临时Tree结构,方便下阶段的聚合。

JOIN::optimize--> 
JOIN::make_tmp_tables_info-->
setup_sum_funcs-->
Item_sum::aggregator_setup-->
Aggregator_simple::setup-->
Item_sum::setup-->

在执行阶段

AggregateIterator::Read()->
reset_and_add()->
aggregator_clear()、aggregator_add()->
Aggregator::clear()、Aggregator::add()->
Item_sum_xxx::add()

在计算distinct聚合时,还需要实现aggregator::endup(),因为distinct_aggregator::add()只是通过某种方式采集了unique的行,但是并未保存,需要在这个阶段进行保存。这个过程可以理解为在distinct的聚合过程中(add)无法判断是否唯一。

注意group by场景下本身是通过临时表解决唯一问题的。

聚合过程(带group by)

MySQL对于带GROUP BY的聚合,通常采用了Temp table的方式保存了(GROUP BY KEY, AGGR VALUE)
具体交给迭代器TemptableAggregateIterator实现。

TemptableAggregateIterator::Init()->
init_tmptable_sum_functions()、update_tmptable_sum_func()->
Item_sum::reset_field()、Item_sum::update_field()->

Item_sum继承于Item_result_field,意味着该类作为计算函数的同时也保存输出结果。具体可以看每个Item_sum子类的val_xxx实现,该函数负责对上层结果或者客户端结果进行输出。

对于特殊聚合函数如AVG\STD\VAR_POP等函数,在累加过程中,临时保存的变量值有多个,实际的输出结果必须通过加工处理,尤其是在group by场景下,多个临时变量需要保存到temp table中,下次累加时取出来,直到最终结果输出。所以需要额外的辅助类Item_result_field,帮助该聚合函数进行最终结果输出。

《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_子查询_06
举例,对于Item_avg_field类的最终结果,需要通过Item_avg_field::val_xxx计算后进行输出。

调用顺序如下:

ExecuteIteratorQuery()->
Query_result_send::send_data()->
THD::send_result_set_row()->
Item::send()->
Item_avg_field::val_xxx

《MySQL 8.0.22执行器源码分析(4.1)Item_sum类以及聚合》_数据库_07
小TIPS:如果内核需要实现多线程并行计算聚合函数的时候,可以通过改造对中间结果输出save_in_field_inner函数,让每个中间结果如按照设计保存到相应的field->ptr中,保留到临时表中。

一些函数(待补全)

Item_sum::check_sum_func

验证聚合函数的语义要求。检查聚合函数的上下文是否允许对其进行聚合,并且当它是另一个聚合函数的参数时,需要直接或间接确保该函数将这两个聚合函数聚合在不同的查询块中。如果将聚合函数聚集在某个外部查询块中,则会将其添加到附加的聚集块查询item链​​inner_sum_func_list​​​中。
tips:解析表达式时,必须为所有的聚合函数调用此函数,而且是以后缀顺序调用。

Item_sum::cleanup

使用后调用每个item。作用是释放所有分配的资源,例如动态内存。通过清除缓存的值为新的执行做准备。它不会删除准备期间分配的值,那些资源由析构函数释放。

参考

​​http://mysql.taobao.org/monthly/2019/05/02/​​​​https://dev.mysql.com/doc/dev/mysql-server/latest/classItem__sum.html​​ MySQL 8.0.22源码


举报

相关推荐

0 条评论