learn.lianglianglee.com/专栏/MySQL实战45讲/36 为什么临时表可以重名?.md.html
2022-08-14 03:40:33 +08:00

452 lines
32 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!DOCTYPE html>
<!-- saved from url=(0046)https://kaiiiz.github.io/hexo-theme-book-demo/ -->
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1.0, user-scalable=no">
<link rel="icon" href="/static/favicon.png">
<title>36 为什么临时表可以重名?.md.html</title>
<!-- Spectre.css framework -->
<link rel="stylesheet" href="/static/index.css">
<!-- theme css & js -->
<meta name="generator" content="Hexo 4.2.0">
</head>
<body>
<div class="book-container">
<div class="book-sidebar">
<div class="book-brand">
<a href="/">
<img src="/static/favicon.png">
<span>技术文章摘抄</span>
</a>
</div>
<div class="book-menu uncollapsible">
<ul class="uncollapsible">
<li><a href="/" class="current-tab">首页</a></li>
</ul>
<ul class="uncollapsible">
<li><a href="../">上一级</a></li>
</ul>
<ul class="uncollapsible">
<li>
<a href="/专栏/MySQL实战45讲/00 开篇词 这一次让我们一起来搞懂MySQL.md.html">00 开篇词 这一次让我们一起来搞懂MySQL</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/01 基础架构一条SQL查询语句是如何执行的.md.html">01 基础架构一条SQL查询语句是如何执行的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/02 日志系统一条SQL更新语句是如何执行的.md.html">02 日志系统一条SQL更新语句是如何执行的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/03 事务隔离:为什么你改了我还看不见?.md.html">03 事务隔离:为什么你改了我还看不见?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/04 深入浅出索引(上).md.html">04 深入浅出索引(上)</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/05 深入浅出索引(下).md.html">05 深入浅出索引(下)</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/06 全局锁和表锁 :给表加个字段怎么有这么多阻碍?.md.html">06 全局锁和表锁 :给表加个字段怎么有这么多阻碍?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/07 行锁功过:怎么减少行锁对性能的影响?.md.html">07 行锁功过:怎么减少行锁对性能的影响?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/08 事务到底是隔离的还是不隔离的?.md.html">08 事务到底是隔离的还是不隔离的?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/09 普通索引和唯一索引,应该怎么选择?.md.html">09 普通索引和唯一索引,应该怎么选择?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/10 MySQL为什么有时候会选错索引.md.html">10 MySQL为什么有时候会选错索引</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/11 怎么给字符串字段加索引?.md.html">11 怎么给字符串字段加索引?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/12 为什么我的MySQL会“抖”一下.md.html">12 为什么我的MySQL会“抖”一下</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/13 为什么表数据删掉一半,表文件大小不变?.md.html">13 为什么表数据删掉一半,表文件大小不变?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/14 count()这么慢,我该怎么办?.md.html">14 count()这么慢,我该怎么办?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/15 答疑文章(一):日志和索引相关问题.md.html">15 答疑文章(一):日志和索引相关问题</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/16 “order by”是怎么工作的.md.html">16 “order by”是怎么工作的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/17 如何正确地显示随机消息?.md.html">17 如何正确地显示随机消息?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/18 为什么这些SQL语句逻辑相同性能却差异巨大.md.html">18 为什么这些SQL语句逻辑相同性能却差异巨大</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/19 为什么我只查一行的语句,也执行这么慢?.md.html">19 为什么我只查一行的语句,也执行这么慢?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/20 幻读是什么,幻读有什么问题?.md.html">20 幻读是什么,幻读有什么问题?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/21 为什么我只改一行的语句,锁这么多?.md.html">21 为什么我只改一行的语句,锁这么多?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/22 MySQL有哪些“饮鸩止渴”提高性能的方法.md.html">22 MySQL有哪些“饮鸩止渴”提高性能的方法</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/23 MySQL是怎么保证数据不丢的.md.html">23 MySQL是怎么保证数据不丢的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/24 MySQL是怎么保证主备一致的.md.html">24 MySQL是怎么保证主备一致的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/25 MySQL是怎么保证高可用的.md.html">25 MySQL是怎么保证高可用的</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/26 备库为什么会延迟好几个小时?.md.html">26 备库为什么会延迟好几个小时?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/27 主库出问题了,从库怎么办?.md.html">27 主库出问题了,从库怎么办?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/28 读写分离有哪些坑?.md.html">28 读写分离有哪些坑?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/29 如何判断一个数据库是不是出问题了?.md.html">29 如何判断一个数据库是不是出问题了?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/30 答疑文章(二):用动态的观点看加锁.md.html">30 答疑文章(二):用动态的观点看加锁</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/31 误删数据后除了跑路,还能怎么办?.md.html">31 误删数据后除了跑路,还能怎么办?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/32 为什么还有kill不掉的语句.md.html">32 为什么还有kill不掉的语句</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/33 我查这么多数据,会不会把数据库内存打爆?.md.html">33 我查这么多数据,会不会把数据库内存打爆?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/34 到底可不可以使用join.md.html">34 到底可不可以使用join</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/35 join语句怎么优化.md.html">35 join语句怎么优化</a>
</li>
<li>
<a class="current-tab" href="/专栏/MySQL实战45讲/36 为什么临时表可以重名?.md.html">36 为什么临时表可以重名?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/37 什么时候会使用内部临时表?.md.html">37 什么时候会使用内部临时表?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/38 都说InnoDB好那还要不要使用Memory引擎.md.html">38 都说InnoDB好那还要不要使用Memory引擎</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/39 自增主键为什么不是连续的?.md.html">39 自增主键为什么不是连续的?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/40 insert语句的锁为什么这么多.md.html">40 insert语句的锁为什么这么多</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/41 怎么最快地复制一张表?.md.html">41 怎么最快地复制一张表?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/42 grant之后要跟着flush privileges吗.md.html">42 grant之后要跟着flush privileges吗</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/43 要不要使用分区表?.md.html">43 要不要使用分区表?</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/44 答疑文章(三):说一说这些好问题.md.html">44 答疑文章(三):说一说这些好问题</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/45 自增id用完怎么办.md.html">45 自增id用完怎么办</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/我的MySQL心路历程.md.html">我的MySQL心路历程</a>
</li>
<li>
<a href="/专栏/MySQL实战45讲/结束语 点线网面一起构建MySQL知识网络.md.html">结束语 点线网面一起构建MySQL知识网络</a>
</li>
</ul>
</div>
</div>
<div class="sidebar-toggle" onclick="sidebar_toggle()" onmouseover="add_inner()" onmouseleave="remove_inner()">
<div class="sidebar-toggle-inner"></div>
</div>
<script>
function add_inner() {
let inner = document.querySelector('.sidebar-toggle-inner')
inner.classList.add('show')
}
function remove_inner() {
let inner = document.querySelector('.sidebar-toggle-inner')
inner.classList.remove('show')
}
function sidebar_toggle() {
let sidebar_toggle = document.querySelector('.sidebar-toggle')
let sidebar = document.querySelector('.book-sidebar')
let content = document.querySelector('.off-canvas-content')
if (sidebar_toggle.classList.contains('extend')) { // show
sidebar_toggle.classList.remove('extend')
sidebar.classList.remove('hide')
content.classList.remove('extend')
} else { // hide
sidebar_toggle.classList.add('extend')
sidebar.classList.add('hide')
content.classList.add('extend')
}
}
function open_sidebar() {
let sidebar = document.querySelector('.book-sidebar')
let overlay = document.querySelector('.off-canvas-overlay')
sidebar.classList.add('show')
overlay.classList.add('show')
}
function hide_canvas() {
let sidebar = document.querySelector('.book-sidebar')
let overlay = document.querySelector('.off-canvas-overlay')
sidebar.classList.remove('show')
overlay.classList.remove('show')
}
</script>
<div class="off-canvas-content">
<div class="columns">
<div class="column col-12 col-lg-12">
<div class="book-navbar">
<!-- For Responsive Layout -->
<header class="navbar">
<section class="navbar-section">
<a onclick="open_sidebar()">
<i class="icon icon-menu"></i>
</a>
</section>
</header>
</div>
<div class="book-content" style="max-width: 960px; margin: 0 auto;
overflow-x: auto;
overflow-y: hidden;">
<div class="book-post">
<p id="tip" align="center"></p>
<div><h1>36 为什么临时表可以重名?</h1>
<p>在上一篇文章中,我们在优化 join 查询的时候使用到了临时表。当时,我们是这么用的:</p>
<pre><code>create temporary table temp_t like t1;
alter table temp_t add index(b);
insert into temp_t select * from t2 where b&gt;=1 and b&lt;=2000;
select * from t1 join temp_t on (t1.b=temp_t.b);
</code></pre>
<p>你可能会有疑问,为什么要用临时表呢?直接用普通表是不是也可以呢?</p>
<p>今天我们就从这个问题说起:临时表有哪些特征,为什么它适合这个场景?</p>
<p>这里,我需要先帮你厘清一个容易误解的问题:有的人可能会认为,临时表就是内存表。但是,这两个概念可是完全不同的。</p>
<ul>
<li>内存表,指的是使用 Memory 引擎的表,建表语法是 create table … engine=memory。这种表的数据都保存在内存里系统重启的时候会被清空但是表结构还在。除了这两个特性看上去比较“奇怪”外从其他的特征上看它就是一个正常的表。</li>
<li>而临时表,可以使用各种引擎类型 。如果是使用 InnoDB 引擎或者 MyISAM 引擎的临时表,写数据的时候是写到磁盘上的。当然,临时表也可以使用 Memory 引擎。</li>
</ul>
<p>弄清楚了内存表和临时表的区别以后,我们再来看看临时表有哪些特征。</p>
<h1>临时表的特性</h1>
<p>为了便于理解,我们来看下下面这个操作序列:</p>
<p><img src="assets/3cbb2843ef9a84ee582330fb1bd0d6e3.png" alt="img" /></p>
<p>图 1 临时表特性示例</p>
<p>可以看到,临时表在使用上有以下几个特点:</p>
<ol>
<li>建表语法是 create temporary table …。</li>
<li>一个临时表只能被创建它的 session 访问,对其他线程不可见。所以,图中 session A 创建的临时表 t对于 session B 就是不可见的。</li>
<li>临时表可以与普通表同名。</li>
<li>session A 内有同名的临时表和普通表的时候show create 语句,以及增删改查语句访问的是临时表。</li>
<li>show tables 命令不显示临时表。</li>
</ol>
<p>由于临时表只能被创建它的 session 访问,所以在这个 session 结束的时候,会自动删除临时表。也正是由于这个特性,<strong>临时表就特别适合我们文章开头的 join 优化这种场景</strong>。为什么呢?</p>
<p>原因主要包括以下两个方面:</p>
<ol>
<li>不同 session 的临时表是可以重名的,如果有多个 session 同时执行 join 优化,不需要担心表名重复导致建表失败的问题。</li>
<li>不需要担心数据删除问题。如果使用普通表,在流程执行过程中客户端发生了异常断开,或者数据库发生异常重启,还需要专门来清理中间过程中生成的数据表。而临时表由于会自动回收,所以不需要这个额外的操作。</li>
</ol>
<h1>临时表的应用</h1>
<p>由于不用担心线程之间的重名冲突,临时表经常会被用在复杂查询的优化过程中。其中,分库分表系统的跨库查询就是一个典型的使用场景。</p>
<p>一般分库分表的场景,就是要把一个逻辑上的大表分散到不同的数据库实例上。比如。将一个大表 ht按照字段 f拆分成 1024 个分表,然后分布到 32 个数据库实例上。如下图所示:</p>
<p><img src="assets/ddb9c43526dfd9b9a3e6f8c153478181.jpg" alt="img" /></p>
<p>图 2 分库分表简图</p>
<p>一般情况下,这种分库分表系统都有一个中间层 proxy。不过也有一些方案会让客户端直接连接数据库也就是没有 proxy 这一层。</p>
<p>在这个架构中,分区 key 的选择是以“减少跨库和跨表查询”为依据的。如果大部分的语句都会包含 f 的等值条件,那么就要用 f 做分区键。这样,在 proxy 这一层解析完 SQL 语句以后,就能确定将这条语句路由到哪个分表做查询。</p>
<p>比如下面这条语句:</p>
<pre><code>select v from ht where f=N;
</code></pre>
<p>这时我们就可以通过分表规则比如N%1024) 来确认需要的数据被放在了哪个分表上。这种语句只需要访问一个分表,是分库分表方案最欢迎的语句形式了。</p>
<p>但是,如果这个表上还有另外一个索引 k并且查询语句是这样的</p>
<pre><code>select v from ht where k &gt;= M order by t_modified desc limit 100;
</code></pre>
<p>这时候,由于查询条件里面没有用到分区字段 f只能到所有的分区中去查找满足条件的所有行然后统一做 order by 的操作。这种情况下,有两种比较常用的思路。</p>
<p>**第一种思路是,**在 proxy 层的进程代码中实现排序。</p>
<p>这种方式的优势是处理速度快,拿到分库的数据以后,直接在内存中参与计算。不过,这个方案的缺点也比较明显:</p>
<ol>
<li>需要的开发工作量比较大。我们举例的这条语句还算是比较简单的,如果涉及到复杂的操作,比如 group by甚至 join 这样的操作,对中间层的开发能力要求比较高;</li>
<li>对 proxy 端的压力比较大,尤其是很容易出现内存不够用和 CPU 瓶颈的问题。</li>
</ol>
<p>**另一种思路就是,**把各个分库拿到的数据,汇总到一个 MySQL 实例的一个表中,然后在这个汇总实例上做逻辑操作。</p>
<p>比如上面这条语句,执行流程可以类似这样:</p>
<ul>
<li>在汇总库上创建一个临时表 temp_ht表里包含三个字段 v、k、t_modified</li>
<li>在各个分库上执行</li>
</ul>
<pre><code>select v,k,t_modified from ht_x where k &gt;= M order by t_modified desc limit 100;
</code></pre>
<ul>
<li>把分库执行的结果插入到 temp_ht 表中;</li>
<li>执行</li>
</ul>
<pre><code>select v from temp_ht order by t_modified desc limit 100;
</code></pre>
<p>得到结果。</p>
<p>这个过程对应的流程图如下所示:</p>
<p><img src="assets/f5ebe0f5af37deeb4d0b63d6fb11fc0d.jpg" alt="img" /></p>
<p>图 3 跨库查询流程示意图</p>
<p>**在实践中,我们往往会发现每个分库的计算量都不饱和,所以会直接把临时表 temp_ht 放到 32 个分库中的某一个上。**这时的查询逻辑与图 3 类似,你可以自己再思考一下具体的流程。</p>
<h1>为什么临时表可以重名?</h1>
<p>你可能会问,不同线程可以创建同名的临时表,这是怎么做到的呢?</p>
<p>接下来,我们就看一下这个问题。</p>
<p>我们在执行</p>
<pre><code>create temporary table temp_t(id int primary key)engine=innodb;
</code></pre>
<p>这个语句的时候MySQL 要给这个 InnoDB 表创建一个 frm 文件保存表结构定义,还要有地方保存表数据。</p>
<p><strong>这个 frm 文件放在临时文件目录下,文件名的后缀是.frm前缀是“#sql{进程 id}<em>{线程 id}</em> 序列号”</strong>。你可以使用 select @@tmpdir 命令,来显示实例的临时文件目录。</p>
<p>而关于表中数据的存放方式,在不同的 MySQL 版本中有着不同的处理方式:</p>
<ul>
<li>在 5.6 以及之前的版本里MySQL 会在临时文件目录下创建一个相同前缀、以.ibd 为后缀的文件,用来存放数据文件;</li>
<li>而从 5.7 版本开始MySQL 引入了一个临时文件表空间,专门用来存放临时文件的数据。因此,我们就不需要再创建 ibd 文件了。</li>
</ul>
<p>从文件名的前缀规则,我们可以看到,其实创建一个叫作 t1 的 InnoDB 临时表MySQL 在存储上认为我们创建的表名跟普通表 t1 是不同的,因此同一个库下面已经有普通表 t1 的情况下,还是可以再创建一个临时表 t1 的。</p>
<p>为了便于后面讨论,我先来举一个例子。</p>
<p><img src="assets/22078eab5c7688c9fbfd6185555bd91b.png" alt="img" /></p>
<p>图 4 临时表的表名</p>
<p>这个进程的进程号是 1234session A 的线程 id 是 4session B 的线程 id 是 5。所以你看到了session A 和 session B 创建的临时表,在磁盘上的文件不会重名。</p>
<p>MySQL 维护数据表,除了物理上要有文件外,内存里面也有一套机制区别不同的表,每个表都对应一个 table_def_key。</p>
<ul>
<li>一个普通表的 table_def_key 的值是由“库名 + 表名”得到的,所以如果你要在同一个库下创建两个同名的普通表,创建第二个表的过程中就会发现 table_def_key 已经存在了。</li>
<li>而对于临时表table_def_key 在“库名 + 表名”基础上又加入了“server_id+thread_id”。</li>
</ul>
<p>也就是说session A 和 sessionB 创建的两个临时表 t1它们的 table_def_key 不同,磁盘文件名也不同,因此可以并存。</p>
<p>在实现上,每个线程都维护了自己的临时表链表。这样每次 session 内操作表的时候,先遍历链表,检查是否有这个名字的临时表,如果有就优先操作临时表,如果没有再操作普通表;在 session 结束的时候,对链表里的每个临时表,执行 “DROP TEMPORARY TABLE + 表名”操作。</p>
<p>这时候你会发现binlog 中也记录了 DROP TEMPORARY TABLE 这条命令。你一定会觉得奇怪,临时表只在线程内自己可以访问,为什么需要写到 binlog 里面?</p>
<p>这,就需要说到主备复制了。</p>
<h1>临时表和主备复制</h1>
<p>既然写 binlog就意味着备库需要。</p>
<p>你可以设想一下,在主库上执行下面这个语句序列:</p>
<pre><code>create table t_normal(id int primary key, c int)engine=innodb;/*Q1*/
create temporary table temp_t like t_normal;/*Q2*/
insert into temp_t values(1,1);/*Q3*/
insert into t_normal select * from temp_t;/*Q4*/
</code></pre>
<p>如果关于临时表的操作都不记录,那么在备库就只有 create table t_normal 表和 insert into t_normal select * from temp_t 这两个语句的 binlog 日志,备库在执行到 insert into t_normal 的时候,就会报错“表 temp_t 不存在”。</p>
<p>你可能会说,如果把 binlog 设置为 row 格式就好了吧?因为 binlog 是 row 格式时,在记录 insert into t_normal 的 binlog 时记录的是这个操作的数据write_row event 里面记录的逻辑是“插入一行数据1,1)”。</p>
<p>确实是这样。如果当前的 binlog_format=row那么跟临时表有关的语句就不会记录到 binlog 里。也就是说,只在 binlog_format=statment/mixed 的时候binlog 中才会记录临时表的操作。</p>
<p>这种情况下,创建临时表的语句会传到备库执行,因此备库的同步线程就会创建这个临时表。主库在线程退出的时候,会自动删除临时表,但是备库同步线程是持续在运行的。所以,这时候我们就需要在主库上再写一个 DROP TEMPORARY TABLE 传给备库执行。</p>
<p>**之前有人问过我一个有趣的问题:**MySQL 在记录 binlog 的时候,不论是 create table 还是 alter table 语句,都是原样记录,甚至于连空格都不变。但是如果执行 drop table t_normal系统记录 binlog 就会写成:</p>
<pre><code>DROP TABLE `t_normal` /* generated by server */
</code></pre>
<p>也就是改成了标准的格式。为什么要这么做呢 </p>
<p>现在你知道原因了那就是drop table 命令是可以一次删除多个表的。比如,在上面的例子中,设置 binlog_format=row如果主库上执行 &quot;drop table t_normal, temp_t&quot;这个命令,那么 binlog 中就只能记录:</p>
<pre><code>DROP TABLE `t_normal` /* generated by server */
</code></pre>
<p>因为备库上并没有表 temp_t将这个命令重写后再传到备库执行才不会导致备库同步线程停止。</p>
<p>所以drop table 命令记录 binlog 的时候,就必须对语句做改写。“/* generated by server */”说明了这是一个被服务端改写过的命令。</p>
<p>说到主备复制,<strong>还有另外一个问题需要解决</strong>:主库上不同的线程创建同名的临时表是没关系的,但是传到备库执行是怎么处理的呢?</p>
<p>现在,我给你举个例子,下面的序列中实例 S 是 M 的备库。</p>
<p><img src="assets/74e789024f10bcde515f21c0368847ba.png" alt="img" /></p>
<p>图 5 主备关系中的临时表操作</p>
<p>主库 M 上的两个 session 创建了同名的临时表 t1这两个 create temporary table t1 语句都会被传到备库 S 上。</p>
<p>但是,备库的应用日志线程是共用的,也就是说要在应用线程里面先后执行这个 create 语句两次。(即使开了多线程复制,也可能被分配到从库的同一个 worker 中执行)。那么,这会不会导致同步线程报错 </p>
<p>显然是不会的,否则临时表就是一个 bug 了。也就是说,备库线程在执行的时候,要把这两个 t1 表当做两个不同的临时表来处理。这,又是怎么实现的呢?</p>
<p>MySQL 在记录 binlog 的时候,会把主库执行这个语句的线程 id 写到 binlog 中。这样,在备库的应用线程就能够知道执行每个语句的主库线程 id并利用这个线程 id 来构造临时表的 table_def_key</p>
<ol>
<li>session A 的临时表 t1在备库的 table_def_key 就是:库名 +t1+“M 的 serverid”+“session A 的 thread_id”;</li>
<li>session B 的临时表 t1在备库的 table_def_key 就是 :库名 +t1+“M 的 serverid”+“session B 的 thread_id”。</li>
</ol>
<p>由于 table_def_key 不同,所以这两个表在备库的应用线程里面是不会冲突的。</p>
<h1>小结</h1>
<p>今天这篇文章,我和你介绍了临时表的用法和特性。</p>
<p>在实际应用中,临时表一般用于处理比较复杂的计算逻辑。由于临时表是每个线程自己可见的,所以不需要考虑多个线程执行同一个处理逻辑时,临时表的重名问题。在线程退出的时候,临时表也能自动删除,省去了收尾和异常处理的工作。</p>
<p>在 binlog_format='row的时候临时表的操作不记录到 binlog 中,也省去了不少麻烦,这也可以成为你选择 binlog_format 时的一个考虑因素。</p>
<p>需要注意的是,我们上面说到的这种临时表,是用户自己创建的 ,也可以称为用户临时表。与它相对应的,就是内部临时表,在[第 17 篇文章]中我已经和你介绍过。</p>
<p>最后,我给你留下一个思考题吧。</p>
<p>下面的语句序列是创建一个临时表,并将其改名:</p>
<p><img src="assets/333ad95b2ce16de1931fe347128caff9.png" alt="img" /></p>
<p>图 6 关于临时表改名的思考题</p>
<p>可以看到,我们可以使用 alter table 语法修改临时表的表名,而不能使用 rename 语法。你知道这是什么原因吗?</p>
<p>你可以把你的分析写在留言区,我会在下一篇文章的末尾和你讨论这个问题。感谢你的收听,也欢迎你把这篇文章分享给更多的朋友一起阅读。</p>
<h1>上期问题时间</h1>
<p>上期的问题是,对于下面这个三个表的 join 语句,</p>
<pre><code>select * from t1 join t2 on(t1.a=t2.a) join t3 on (t2.b=t3.b) where t1.c&gt;=X and t2.c&gt;=Y and t3.c&gt;=Z;
</code></pre>
<p>如果改写成 straight_join要怎么指定连接顺序以及怎么给三个表创建索引。</p>
<p>第一原则是要尽量使用 BKA 算法。需要注意的是,使用 BKA 算法的时候,并不是“先计算两个表 join 的结果,再跟第三个表 join”而是直接嵌套查询的。</p>
<p>具体实现是:在 t1.c&gt;=X、t2.c&gt;=Y、t3.c&gt;=Z 这三个条件里,选择一个经过过滤以后,数据最少的那个表,作为第一个驱动表。此时,可能会出现如下两种情况。</p>
<p>第一种情况,如果选出来是表 t1 或者 t3那剩下的部分就固定了。</p>
<ol>
<li>如果驱动表是 t1则连接顺序是 t1-&gt;t2-&gt;t3要在被驱动表字段创建上索引也就是 t2.a 和 t3.b 上创建索引;</li>
<li>如果驱动表是 t3则连接顺序是 t3-&gt;t2-&gt;t1需要在 t2.b 和 t1.a 上创建索引。</li>
</ol>
<p>同时,我们还需要在第一个驱动表的字段 c 上创建索引。</p>
<p>第二种情况是,如果选出来的第一个驱动表是表 t2 的话,则需要评估另外两个条件的过滤效果。</p>
<p>总之,整体的思路就是,尽量让每一次参与 join 的驱动表的数据集,越小越好,因为这样我们的驱动表就会越小。</p>
</div>
</div>
<div>
<div style="float: left">
<a href="/专栏/MySQL实战45讲/35 join语句怎么优化.md.html">上一页</a>
</div>
<div style="float: right">
<a href="/专栏/MySQL实战45讲/37 什么时候会使用内部临时表?.md.html">下一页</a>
</div>
</div>
</div>
</div>
</div>
</div>
<a class="off-canvas-overlay" onclick="hide_canvas()"></a>
</div>
<script defer src="https://static.cloudflareinsights.com/beacon.min.js/v652eace1692a40cfa3763df669d7439c1639079717194" integrity="sha512-Gi7xpJR8tSkrpF7aordPZQlW2DLtzUlZcumS8dMQjwDHEnw9I7ZLyiOj/6tZStRBGtGgN6ceN6cMH8z7etPGlw==" data-cf-beacon='{"rayId":"709972dcabc93d60","version":"2021.12.0","r":1,"token":"1f5d475227ce4f0089a7cff1ab17c0f5","si":100}' crossorigin="anonymous"></script>
</body>
<!-- Global site tag (gtag.js) - Google Analytics -->
<script async src="https://www.googletagmanager.com/gtag/js?id=G-NPSEEVD756"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag() {
dataLayer.push(arguments);
}
gtag('js', new Date());
gtag('config', 'G-NPSEEVD756');
var path = window.location.pathname
var cookie = getCookie("lastPath");
console.log(path)
if (path.replace("/", "") === "") {
if (cookie.replace("/", "") !== "") {
console.log(cookie)
document.getElementById("tip").innerHTML = "<a href='" + cookie + "'>跳转到上次进度</a>"
}
} else {
setCookie("lastPath", path)
}
function setCookie(cname, cvalue) {
var d = new Date();
d.setTime(d.getTime() + (180 * 24 * 60 * 60 * 1000));
var expires = "expires=" + d.toGMTString();
document.cookie = cname + "=" + cvalue + "; " + expires + ";path = /";
}
function getCookie(cname) {
var name = cname + "=";
var ca = document.cookie.split(';');
for (var i = 0; i < ca.length; i++) {
var c = ca[i].trim();
if (c.indexOf(name) === 0) return c.substring(name.length, c.length);
}
return "";
}
</script>
</html>