回表
Mysql回表指的是在InnoDB存储引擎下,二级索引查询到的索引列,如果需要查找所有列的数据,则需要到主键索引里面去取出数据。这个过程就称为回表。因为行的数据都是存在主键B+tree的叶子节点里面,二级索引的B+树叶子节点都是存放的(索引列,主键)。
例如:
有一张用户表 id(主键), username (二级索引),age
id是主键索引 username是二级索引,当我们通过二级索引查询返回记录时,例如:
select * from t where username = “bobo”;
① 通过二级索引找到 bobo记录和主键id
② 在通过主键id去主键索引B+tree找到行记录
这里就会出现一个问题,当我们的用户表有150w条记录时, 有时候查询并没有走索引,而是通过All全盘的扫描。这是为什么了?下面我们就简单的介绍一下全盘扫描和走索引的IO。
假如我们一行数据的大小是100字节, bobo的有50W行数据. 假设用户表的B+树高度=3
通过username索引的话:每次回表需要3次IO,50w记录需要回表50w次。那么IO = 50W * 3
通过全盘扫描:一个页可以存放16k / 100字节 = 16 * 1024 / 100 = 163条记录 全盘扫描则需要 150w / 163 = 0.92W 差不多是1w次IO,
通过上面对比,肯定全盘要效率高一些,这就是为什么有时候我们建立了索引,但是通过explain检测的时候我们的sql没有走索引。因为回表的IO开销还是比较大的。