1.什么是临时表

内部临时表是sql语句执行过程中,用来存储中间结果的的数据表,其作用类似于:join语句执行过程中的joinbuffer,order by语句执行过程中的sortBuffer一样。这个表是mysql自己创建出来的,对客户端程序不可见。那么mysql什么时候会创建内部临时表呢?创建的内部临时表的表结构又是怎么样的呢?

2.临时表的使用场景

在mysql中常见的使用临时表的场景,有两个:unoin语句和groupby语句。

为了更好的了解内部临时表在unoin和groupby中是如何起作用的,我们先了解一下unoin和groupby的执行流程。

为了方便下文的描述,我们建立如下表结构:

CREATE TABLE `t1` (`id` int(11) NOT NULL,`a` int(11)  DEFAULT NULL,`b` int(11) default null,PRIMARY KEY (`id`) USING BTREE,key (`a`) using BTREE
) ENGINE=InnoDB;

建立表t1,其中id为主键,a为普通索引,然后向表中插入1000条数据:

drop procedure idata;
delimiter ;;
create procedure idata()
begindeclare i int;set i=1;while(i<=1000)doinsert into `t1` values(i,i,i);set i=i+1;end while;
end;;
delimiter ;
call idata();

union

我们都知道union的语义是对 unoin两端的结果集取并集,也就是两个结果集加起来,重复的数据行,只取其中一行。这里需要注意,unoin是有在多个数据集中排重的语义的。

下面我们执行下面这条语句:

(select 1000 as f) union (select id from t1 order by id desc limit 2);

在这条语句的语义是:将t1中的数据,按照id倒序排列后,取出前两行数据的id,与"1000"取并集。

这条语句在mysql中的执行流程如下:

1.创建一个内存临时表,这个内存临时表只有一个整形字段f,并且f字段为主键(因为要进行排重)。

2.执行第一个子查询,得到1000这个值,放入到内存临时表中。

3.执行第二个子查询:取出第一个满足条件数据行中的id=1000,尝试写入临时表,这时会出现违反唯一性约束的情况,导致插入失败,然后继续执行。取出第二行数据的id=999,插入成功。

4.从临时表中取出数据,返回客户端结果,并删除临时表。

同时,我们可以查看上述查询语句的执行计划,来验证上述执行流程:

从以上过程中,我们可以知道,内存临时表的作用:通过唯一键约束,实现了union的语义。

如果把上述语句中的union,替换成 union all的话,那查询语句就失去了"去重"的语义了。那么,mysql在执行查询语句的过程中,是否还会使用临时表呢?

我们使用以下查询语句进行验证:

(select 1000 as f) union all (select id from t1 order by id desc limit 2);

通过查询sql的执行计划,我们会发现,查询语句执行过程中,不在需要临时表了。

整个查询语句的执行流程如下:

1.执行第一个子查询,将查询的结果,作为结果集的一部分,返回给客户端。

2.执行第二个子查询,将查询的结果作为结果集的一部分,返回给客户端。

groupby

除了unoin查询语句在执行过程中会使用临时表外,groupby 查询语句在执行过程中,也会使用临时表。为了方便说明问题,我们执行如下查询语句:

select id%10 as m, count(1) as c from t1 group by m order by m;

该语句的语义,将表中所有数据中的id值,对10进行取模,并将取模后的结果进行分组,然后统计出每组数据的个数。查询语句执行计划如下:

该查询语句的执行流程如下:

1.创建临时表,表中有两个字段:m和c,其中m为主键,因为group by字段m的值,必须是唯一的。

2.扫描表t1的索引a,依次取出叶子结点上的id值,并计算id%10,将计算结果记为x,如果临时表中没有m=x的行,就插入一个记录(x,1)。如果表中有m=x数据行,那么就将x这一行的c值加1。

3.遍历完成后,在根据字段m做排序,得到最终结果返回给客户端。

对于步骤3中的排序流程,可以参考 如何优化sql中的orderBy。

3.groupby 如何优化

通过上面的描述,我们知道了groupby的执行流程。groupby在执行过程中,需要建立一个带有唯一键索引的临时表,其中唯一键索引字段就是groupby的字段。这个执行代价还是比较高的,而且这个临时表还是一次性的。

为了提高groupby语句的执行性能,我们可以从"不使用临时表"的角度下手。首先我们可以这样想:要想让groupby的过程中不使用临时表,我们就要知道,临时表在groupby的过程中,解决了什么问题?如果,我们能找到另外一种不使用临时表,也能解决这个问题的方案,那么我们就可以不使用临时表了。

首先,我们知道,在日常开发过程中,我们使用groupby主要就是为了实现:将表中所有的数据,按照指定字段进行分组。把字段值相同的数据划分为一个组,然后在对组内的数据执行聚合函数,聚合函数计算的结果,作为结果集中的一行数据。

而在这个过程中,临时表的作用就是在扫描数据表的时候,对每行数据属于哪个组,进行记录,同时执行聚合函数的逻辑。之所以需要一个临时表来记录每行数据属于哪个组,主要是因为表中的数据,按照"group by字段"维度,不是有序的。

如果表中的数据本身就是按照"groupby字段"有序的话,也就是属于同一个组的数据都分布在一起,那么就不需要临时表,也可以对数据进行分组。 举例如下图,如果执行groupby,同时计算每组数据个数。执行流程大致如下:

1.从左到右扫描数据,并依次累加,当遇到第一个2时,说明已经积累了3个1了,此时结果集的第一行数据就是(1,3)。

2.当遇到第一个3的时候,说明已经积累了2个2了,此时结果集的第二行数据就是(2,2);

3.按照以上逻辑逐个计算,就可以得到最终结果。

在mysql中,如果分组字段上有索引的话,执行查询过程中,mysql就不会建立临时表了。
我们可以执行如下查询语句进行验证:

explain
select id as m from t1 group by id;

通过查看执行计划,我们可以发现,因为分组字段id,是主键,本身是有序的。这里并没有使用临时表:

但是很多时候,分组字段并不是表中的一个具体字段。而是通过一定计算后的逻辑字段,如:

select id%10 as m from t1 group by m

这里分组字段m,并不是t1表中的一个字段,而是对id对10取模后的一个逻辑字段。为了让分组字段有序,下面给大家介绍两种优化手段。

1.生成伴生字段,并建立索引

从mysql 5.7开始,支持了generated column机制,来实现字段数据的关联更新。如下语句:

alter table t1 add column z int generated always as (id % 10), add index (z);

为t1表增加字段z,z的字段值为id值与10取模后的结果,同时在z上添加索引。这样当我们再执行:

explain
select id%100 as m,count(*) as c from t1 group by m

或者:

explain
select z as m,count(id) as c from t1 group by m

执行计划如下:

此时就不在使用临时表了。

上面的伴生字段的方案,需要我们向表中添加额外字段,如果业务场景比较复杂,分组的场景比较多,使用伴生字段方案需要在表中增加的额外字段就会比较多。这将会使我们的数据表结果变得比较复杂。

2.直接对分组字段进行排序

如果我们可以预估到,在执行groupby语句时,分组后的数据量比较大,使用的内存临时表可能都无法存储,那么内存临时表就会被替换成磁盘临时表,这个替换的阈值,由变量"tmp_table_size"控制,该变量的默认值为16M,如果在查询语句执行过程,需要存放到临时表中的数据量超过16M,那么使用的临时表就会变成磁盘临时表,磁盘临时表默认的存储引擎是InnoDB,磁盘临时表的性能相比内存临时表性能更低。

对于这种情况,mysql提供了 SQL_BIG_RESULT语句,该语句的作用就是告诉优化器:这个语句涉及到的数据量比较大,直接使用磁盘临时表。但是这里使用的磁盘临时表,会调整存储的数据结构,数据结构不再是B+树,而是数组。

下面我们举例说明,执行如下查询语句的的流程如下:

explain
select sql_big_result id%100 as m,count(id) as c from t1 group by m ;

执行流程:
1.初始化sort_buffer,确定放入一个整形字段,记为m。

2.扫描t1索引a,依次取出叶子节点中的主键id的值,并对100取模,然后插入到sort_buffer中。

3.数据表扫描完后,对sort_buffer中的m进行排序。

4.排序后,就得到了一个针对分组字段的有序数组。

有了针对分组字段的有序数组,那么就可以通过遍历该数组实现groupby的语义了。

通过查看上述查询语句的执行计划,可以发现,不在使用临时表了。

总结

为了保证groupby的执行性能,在使用groupby的时候要做到以下几点:

1.尽量让 group by 过程用上表的索引,确认方法是 explain 结果里没有 Using temporary 和 Using filesort。

2.如果 group by 需要统计的数据量不大,尽量只使用内存临时表;也可以通过适当调大 tmp_table_size 参数,来避免用到磁盘临时表。

3.如果数据量实在太大,使用 SQL_BIG_RESULT 这个提示,来告诉优化器直接使用排序算法得到 group by 的结果。

mysql中的临时表怎么用的?相关推荐

  1. mysql 临时列_如何在MySQL中列出临时表列?

    要列出MySQL中的临时表列,让我们首先创建一个临时表. 这是一个例子.我们创建了一个临时表,其中包含一些列,其中包括学生的详细信息-mysql> CREATE TEMPORARY TABLE  ...

  2. MySQL中关于临时表的创建到删除详细过程

    1.临时表的创建: CREATE TEMPORARY TABLE SalesSummary (product_name VARCHAR(50) NOT NULL, total_sales DECIMA ...

  3. mysql 过程 临时表_在存储过程mysql中创建临时表

    我有SQL Server的经验.这是我第一次使用mysql.我想在存储过程中创建一个临时表.我不知道我在这里错过了什么. 我想做的是: 循环遍历事件及其匹配项,并将这些匹配项插入到临时表中,然后从该临 ...

  4. mysql 创建临时表权限_有没有更好的方法为MySQL中的临时表分配权限?

    我们的用户以相当低级别的用户身份登录到生产数据库,在数据库级别授予SELECT,并在他们需要访问的特定表上授予INSERT / UPDATE / DELETE. 他们还有权创建临时表(我们需要它们来处 ...

  5. MySQL中查询字段为空或者为null方法

    MySQL中查询字段为空或者为null方法 判断为null select * from table where column is null; 不为null: select * from table ...

  6. mysql reopen table_mysql 临时表 cann't reopen解决方案

    当你创建临时表的时候,你可以使用temporary关键字.如:复制代码 代码如下:create temporary table tmp_table(name varchar(10) not null, ...

  7. mysql如何创建临时表

    在 MySQL 中创建临时表的方法如下: 在你要创建的临时表的 SELECT 语句之前,使用 CREATE TEMPORARY TABLE 语句来创建临时表,格式如下: CREATE TEMPORAR ...

  8. mysql 避免使用临时表_从日期范围中选择时防止MySQL使用临时表

    我试图阻止mysql在此查询中创建临时表 SELECT `vendor_id`,SUM(`qty`) AS `qty` FROM `inventory_transactions` WHERE `inv ...

  9. MySQL 中的myisam内部临时表

    摘要: 本文只是记录一下验证过程,源码比较复杂,时间有限没仔细读过.如有误导请见谅. 源码版本 percona 5.7.14 一.问题由来 一个朋友问我下面的tmp目录的文件是干什么的,一会就删除了. ...

最新文章

  1. 面向隐私AI的TensorFlow深度定制化实践
  2. MPB:中科院城环所杨军组-​​淡水浮游动物的采集及鉴定
  3. 高性能ASP.NET站点构建之托管资源优化
  4. 【C / C++】关于数组默认初值问题
  5. Kubernetes初步了解及入门
  6. bitnamigitlab_Bitnami Gitlab 安装配置 step by step
  7. [回归分析][10]--相关误差的问题
  8. mysql.ini环境配置_MySQL配置文件mysql.ini参数详解
  9. 有赞再推视频号流量扶持政策 单商家单月最高可获5万流量奖励
  10. python进阶20装饰器
  11. 结合CDIB类,对图像的打开、显示、保存
  12. 接口测试-jmeter
  13. 【VBA】Excel 密码管理器
  14. 最新最火最流行的抖音火山上热门技术!
  15. 怎样找回通讯录的联系人号码?手机通信录联系人恢复教程推荐
  16. Kafka【问题 02】KafkaTemplate 报错 Bootstrap broker localhost:9092 (id: -1 rack: null) disconnected 问题解决
  17. java如何使用conver_Springmvc conver实现原理及用法解析
  18. 干货干货~C语言版学生成绩管理系统【数据结构课程设计,百行代码实现功能强化版(内附源码)】
  19. 313day(服务器的一些问题)
  20. java框架013——Spring AOP面向切面编程

热门文章

  1. airtest获取当前设备序列号,并连接
  2. 量子计算机怎么储存,什么是量子计算机_量子计算机原理_量子计算的两种有效方法...
  3. 基于spss的多元线性回归(逐步回归法 stepwise regression)
  4. 观世音菩萨为什么不直接将整个地狱的众生都拯救出来?
  5. hive建表报错FAILED: ParseException line 2:0 Failed to recognize predicate ‘date‘. Failed rule: ‘identifi
  6. 网络存储技术Windows server 2012(项目三 存储池的配置与管理)
  7. 运营人员必知!SPU和SKU是什么?
  8. 国际海运出口的操作流程是怎样的?
  9. 功放限幅保护_一种功放限幅器的制作方法
  10. 【图像检测】基于计算机视觉实现地质断层结构的自动增强和识别附matlab代码