MySQL 内部临时表

本文详细解析了MySQL中内部临时表的使用,包括UNION操作如何使用临时表去重,GROUP BY语句在内存充足和不足时的执行过程,以及优化策略。讨论了ORDER BY NULL、内存不足时的优化方案,如优化索引和直接排序,并对比了DISTINCT与GROUP BY的执行性能。最后总结了何时会用到内部临时表,并提供了避免临时表使用的建议。
摘要由CSDN通过智能技术生成

UNION

UNION语义:取两个子查询结果的并集,重复的行只保留一行

表初始化

CREATE TABLE t1(id INT PRIMARY KEY, a INT, b INT, INDEX(a));
DELIMITER ;;
CREATE PROCEDURE idata()
BEGIN
    DECLARE i INT;

    SET i=1;
    WHILE (i<= 1000) DO
        INSERT INTO t1 VALUES (i,i,i);
        SET i=i+1;
    END WHILE;
END;;
DELIMITER ;
CALL idata();

执行语句

(SELECT 1000 AS f) UNION (SELECT id FROM t1 ORDER BY id DESC LIMIT 2);

mysql> EXPLAIN (SELECT 1000 AS f) UNION (SELECT id FROM t1 ORDER BY id DESC LIMIT 2);
+----+--------------+------------+------------+-------+---------------+---------+---------+------+------+----------+----------------------------------+
| id | select_type  | table      | partitions | type  | possible_keys | key     | key_len | ref  | rows | filtered | Extra                            |
+----+--------------+------------+------------+-------+---------------+---------+---------+------+------+----------+----------------------------------+
|  1 | PRIMARY      | NULL       | NULL       | NULL  | NULL          | NULL    | NULL    | NULL | NULL |     NULL | No tables used                   |
|  2 | UNION        | t1         | NULL       | index | NULL          | PRIMARY | 4       | NULL |    2 |   100.00 | Backward index scan; Using index |
| NULL | UNION RESULT | <union1,2> | NULL       | ALL   | NULL          | NULL    | NULL    | NULL | NULL |     NULL | Using temporary                  |
+----+--------------+------------+------------+-------+---------------+---------+---------+------+------+----------+----------------------------------+
  1. 第二行的Key=PRIMARY,说明第二个子查询用到了索引id
  2. 第三行的Extra字段为Using temporary。 表示在对子查询的结果做UNION RESULT的时候,使用了临时表

union result

在这里插入图片描述

  1. 创建一个内存临时表,这个内存临时表只有一个整型字段f,并且f为主键
  2. 执行第一个子查询,得到1000,并存入内存临时表中
  3. 执行第二个子查询
    1. 拿到第一行id=1000,试图插入到内存临时表,但由于1000这个值已经存在于内存临时表。违反唯一性约束,插入失败,继续执行
    2. 拿到第二行id=999,插入内存临时表成功
  4. 从内存临时表中按行取出数据,返回结果,并删除内存临时表,结果中包含id=1000和id=999两行
  5. 内存临时表起到了暂存数据的作用,还用到了内存临时表主键id的唯一性约束,实现UNION的语义

UNION ALL

UNION ALL没有去重的语义,一次执行子查询,得到的结果直接发给客户端,不需要内存临时表

mysql> EXPLAIN (SELECT 1000 AS f) UNION ALL (SELECT id FROM t1 ORDER BY id DESC LIMIT 2);
+----+----------
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值