「秋招系列」MySQL 面试核心 25 问(附答案)

1、为什么使用索引?

  • 通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。

  • 可以大大加快数据的检索速度,这也是创建索引的最主要的原因。

  • 帮助服务器避免排序和临时表

  • 将随机 IO 变为顺序 IO。

  • 可以加速表和表之间的连接,特别是在实现数据的参考完整性方面特别有意义。

2、Innodb 为什么要用自增 id 作为主键?

如果表使用自增主键,那么每次插入新的记录,记录就会顺序添加到当前索引节点的后续位置,当一页写满,就会自动开辟一个新的页。如果使用非自增主键(如果身份证号或学号等),由于每次插入主键的值近似于随机,因此每次新纪录都要被插到现有索引页得中间某个位置, 频繁的移动、分页操作造成了大量的碎片,得到了不够紧凑的索引结构,后续不得不通过 OPTIMIZE TABLE(optimize table)来重建表并优化填充页面。

3、MyISAM 和 InnoDB 实现 B 树索引方式的区别是什么?

  • MyISAM,B+Tree 叶节点的 data 域存放的是数据记录的地址,在索引检索的时候,首先按照 B+Tree 搜索算法搜索索引,如果指定的 key 存在,则取出其 data 域的值,然后以 data 域的值为地址读取相应的数据记录,这被称为“非聚簇索引”

  • InnoDB,其数据文件本身就是索引文件,相比 MyISAM,索引文件和数据文件是分离的,其表数据文件本身就是按 B+Tree 组织的一个索引结构,树的节点 data 域保存了完整的数据记录,这个索引的 key 是数据表的主键,因此 InnoDB 表数据文件本身就是主索引,这被称为“聚簇索引”或者聚集索引,而其余的索引都作为辅助索引,辅助索引的 data 域存储相应记录主键的值而不是地址,这也是和 MyISAM 不同的地方。在根据主索引搜索时,直接找到 key 所在的节点即可取出数据;在根据辅助索引查找时,则需要先取出主键的值,再走一遍主索引。因此,在设计表的时候,不建议使用过长的字段为主键,也不建议使用非单调的字段作为主键,这样会造成主索引频繁分裂。

4、说一下 MySQL 是如何执行一条 SQL 的?具体步骤有哪些?

SQL 执行的全部过程

Server 层按顺序执行 sql 的步骤为:

  1. 客户端请求->

  2. 连接器(验证用户身份,给予权限) ->

  3. 查询缓存(存在缓存则直接返回,不存在则执行后续操作)->

  4. 分析器(对 SQL 进行词法分析和语法分析操作) ->

  5. 优化器(主要对执行的 sql 优化选择最优的执行方案方法) ->

  6. 执行器(执行时会先看用户是否有执行权限,有才去使用这个引擎提供的接口)->

  7. 去引擎层获取数据返回(如果开启查询缓存则会缓存查询结果)

简单概括:

  • 连接器:管理连接、权限验证;

  • 查询缓存:命中缓存则直接返回结果;

  • 分析器:对 SQL 进行词法分析、语法分析;(判断查询的 SQL 字段是否存在也是在这步)

  • 优化器:执行计划生成、选择索引;

  • 执行器:操作引擎、返回结果;

  • 存储引擎:存储数据、提供读写接口。

5、你了解 MySQL 的内部构造吗?一般可以分为哪两个部分?

可以分为服务层和存储引擎层两部分,其中:

服务层包括连接器、查询缓存、分析器、优化器、执行器等,涵盖 MySQL 的大多数核心服务功能,以及所有的内置函数(如日期、时间、数学和加密函数等),所有跨存储引擎的功能都在这一层实现,比如存储过程、触发器、视图等。

存储引擎层负责数据的存储和提取。其架构模式是插件式的,支持 InnoDB、MyISAM、Memory 等多个存储引擎。现在最常用的存储引擎是 InnoDB,它从 MySQL 5.5.5 版本开始成为了默认的存储引擎。

6、说一说 Drop、Delete 与 Truncate 的共同点和区别

第一种回答

Drop、Delete、Truncate 都表示删除,但是三者有一些差别:

Delete 用来删除表的全部或者一部分数据行,执行 Delete 之后,用户需要提交(commmit)或者回滚(rollback)来执行删除或者撤销删除,会触发这个表上所有的 delete 触发器。

Truncate 删除表中的所有数据,这个操作不能回滚,也不会触发这个表上的触发器,TRUNCATE 比 Delete 更快,占用的空间更小。

Drop 命令从数据库中删除表,所有的数据行,索引和权限也会被删除,所有的 DML 触发器也不会被触发,这个命令也不能回滚。

因此,在不再需要一张表的时候,用 Drop;在想删除部分数据行时候,用 Delete;在保留表而删除所有数据的时候用 Truncate。

第二种回答

  • Drop 直接删掉表;

  • Truncate 删除表中数据,再插入时自增长 id 又从 1 开始 ;

  • Delete 删除表中数据,可以加 where 字句。

具体解析

  1. DELETE 语句执行删除的过程是每次从表中删除一行,并且同时将该行的删除操作作为事务记录在日志中保存以便进行进行回滚操作。TRUNCATE TABLE 则一次性地从表中删除所有的数据并不把单独的删除操作记录记入日志保存,删除行是不能恢复的。并且在删除的过程中不会激活与表有关的删除触发器。执行速度快。

  2. 表和索引所占空间。当表被 TRUNCATE 后,这个表和索引所占用的空间会恢复到初始大小,而 DELETE 操作不会减少表或索引所占用的空间。drop 语句将表所占用的空间全释放掉。

  3. 一般而言,drop > truncate > delete

  4. 应用范围。TRUNCATE 只能对 TABLE;DELETE 可以是 table 和 view

  5. TRUNCATE 和 DELETE 只删除数据,而 DROP 则删除整个表(结构和数据)。

  6. truncate 与不带 where 的 delete :只删除数据,而不删除表的结构(定义)drop 语句将删除表的结构被依赖的约束(constrain),触发器(trigger)索引(index);依赖于该表的存储过程/函数将被保留,但其状态会变为:invalid。

  7. delete 语句为 DML(Data Manipulation Language),这个操作会被放到 rollback segment 中,事务提交后才生效。如果有相应的 tigger,执行的时候将被触发。

  8. truncate、drop 是 DDL(Data Define Language),操作立即生效,原数据不放到 rollback segment 中,不能回滚

  9. 在没有备份情况下,谨慎使用 drop 与 truncate。要删除部分数据行采用 delete 且注意结合 where 来约束影响范围。回滚段要足够大。要删除表用 drop;若想保留表而将表中数据删除,如果与事务无关,用 truncate 即可实现。如果和事务有关,或老是想触发 trigger,还是用 delete。

  10. Truncate table 表名 速度快,而且效率高,因为: truncate table 在功能上与不带 WHERE 子句的 DELETE 语句相同:二者均删除表中的全部行。但 TRUNCATE TABLE 比 DELETE 速度快,且使用的系统和事务日志资源少。DELETE 语句每次删除一行,并在事务日志中为所删除的每行记录一项。TRUNCATE TABLE 通过释放存储表数据所用的数据页来删除数据,并且只在事务日志中记录页的释放。

  11. TRUNCATE TABLE 删除表中的所有行,但表结构及其列、约束、索引等保持不变。新行标识所用的计数值重置为该列的种子。如果想保留标识计数值,请改用 DELETE。如果要删除表定义及其数据,请使用 DROP TABLE 语句。

  12. 对于由 FOREIGN KEY 约束引用的表,不能使用 TRUNCATE TABLE,而应使用不带 WHERE 子句的 DELETE 语句。由于 TRUNCATE TABLE 不记录在日志中,所以它不能激活触发器。

7、MySQL 优化了解吗?说一下从哪些方面可以做到性能优化?

  • 为搜索字段创建索引

  • 避免使用 Select *,列出需要查询的字段

  • 垂直分割分表

  • 选择正确的存储引擎

8、数据库隔离级别

  • 未提交读,事务中发生了修改,即使没有提交,其他事务也是可见的,比如对于一个数 A 原来 50 修改为 100,但是我还没有提交修改,另一个事务看到这个修改,而这个时候原事务发生了回滚,这时候 A 还是 50,但是另一个事务看到的 A 是 100.可能会导致脏读、幻读或不可重复读

  • 提交读,对于一个事务从开始直到提交之前,所做的任何修改是其他事务不可见的,举例就是对于一个数 A 原来是 50,然后提交修改成 100,这个时候另一个事务在 A 提交修改之前,读取的 A 是 50,刚读取完,A 就被修改成 100,这个时候另一个事务再进行读取发现 A 就突然变成 100 了;可以阻止脏读,但是幻读或不可重复读仍有可能发生

  • 重复读,就是对一个记录读取多次的记录是相同的,比如对于一个数 A 读取的话一直是 A,前后两次读取的 A 是一致的;可以阻止脏读和不可重复读,但幻读仍有可能发生

  • 可串行化读,在并发情况下,和串行化的读取的结果是一致的,没有什么不同,比如不会发生脏读和幻读;该级别可以防止脏读、不可重复读以及幻读

隔离级别脏读不可重复读幻影读 READ-UNCOMMITTED 未提交读√√√READ-COMMITTED 提交读×√√REPEATABLE-READ 重复读××√SERIALIZABLE 可串行化读×××

MySQL InnoDB 存储引擎的默认支持的隔离级别是 REPEATABLE-READ(可重读)

这里需要注意的是:与 SQL 标准不同的地方在于 InnoDB 存储引擎在 REPEATABLE-READ(可重读)事务隔离级别 下使用的是 Next-Key Lock 锁算法,因此可以避免幻读的产生,这与其他数据库系统(如 SQL Server)是不同的。所以 说 InnoDB 存储引擎的默认支持的隔离级别是 REPEATABLE-READ(可重读) 已经可以完全保证事务的隔离性要 求,即达到了 SQL 标准的 SERIALIZABLE(可串行化)隔离级别。

因为隔离级别越低,事务请求的锁越少,所以大部分数据库系统的隔离级别都是 READ-COMMITTED(读取提交内 容):,但是你要知道的是 InnoDB 存储引擎默认使用 REPEATABLE-READ(可重读)并不会有任何性能损失

InnoDB 存储引擎在分布式事务 的情况下一般会用到 SERIALIZABLE(可串行化)隔离级别。

9、都知道数据库索引采用 B+树而不是 B 树,原因也有很多,主要原因是什么?

主要原因:B+树只要遍历叶子节点就可以实现整棵树的遍历,而且在数据库中基于范围的查询是非常频繁的,而 B 树只能中序遍历所有节点,效率太低。

10、文件索引和数据库索引为什么使用 B+树?(第 9 个问题的详细回答)

文件与数据库都是需要较大的存储,也就是说,它们都不可能全部存储在内存中,故需要存储到磁盘上。而所谓索引,则为了数据的快速定位与查找,那么索引的结构组织要尽量减少查找过程中磁盘 I/O 的存取次数,因此 B+树相比 B 树更为合适。数据库系统巧妙利用了局部性原理与磁盘预读原理,将一个节点的大小设为等于一个页,这样每个节点只需要一次 I/O 就可以完全载入,而红黑树这种结构,高度明显要深的多,并且由于逻辑上很近的节点(父子)物理上可能很远,无法利用局部性。

最重要的是,B+树还有一个最大的好处:方便扫库。

B 树必须用中序遍历的方法按序扫库,而 B+树直接从叶子结点挨个扫一遍就完了,B+树支持 range-query 非常方便,而 B 树不支持,这是数据库选用 B+树的最主要原因。

B+树查找效率更加稳定,B 树有可能在中间节点找到数据,稳定性不够。

B+tree 的磁盘读写代价更低:B+tree 的内部结点并没有指向关键字具体信息的指针(红色部分),因此其内部结点相对 B 树更小。如果把所有同一内部结点的关键字存放在同一块盘中,那么盘块所能容纳的关键字数量也越多。一次性读入内存中的需要查找的关键字也就越多,相对来说 IO 读写次数也就降低了;

B+tree 的查询效率更加稳定:由于内部结点并不是最终指向文件内容的结点,而只是叶子结点中关键字的索引,所以,任何关键字的查找必须走一条从根结点到叶子结点的路。所有关键字查询的路径长度相同,导致每一个数据的查询效率相当;

11、听说过视图吗?那游标呢?

视图是一种虚拟的表,通常是有一个表或者多个表的行或列的子集,具有和物理表相同的功能 游标是对查询出来的结果集作为一个单元来有效的处理。一般不使用游标,但是需要逐条处理数据的时候,游标显得十分重要。

12、MySQL 中为什么要有事务回滚机制?

而在 MySQL 中,恢复机制是通过回滚日志(undo log)实现的,所有事务进行的修改都会先记录到这个回滚日志中,然后在对数据库中的对应行进行写入。当事务已经被提交之后,就无法再次回滚了。

回滚日志作用:1)能够在发生错误或者用户执行 ROLLBACK 时提供回滚相关的信息 2) 在整个系统发生崩溃、数据库进程直接被杀死后,当用户再次启动数据库进程时,还能够立刻通过查询回滚日志将之前未完成的事务进行回滚,这也就需要回滚日志必须先于数据持久化到磁盘上,是我们需要先写日志后写数据库的主要原因。

13、数据库引擎 InnoDB 与 MyISAM 的区别

InnoDB

  • 是 MySQL 默认的事务型存储引擎,只有在需要它不支持的特性时,才考虑使用其它存储引擎。

  • 实现了四个标准的隔离级别,默认级别是可重复读(REPEATABLE READ)。在可重复读隔离级别下,通过多版本并发控制(MVCC)+ 间隙锁(Next-Key Locking)防止幻影读。

  • 主索引是聚簇索引,在索引中保存了数据,从而避免直接读取磁盘,因此对查询性能有很大的提升。

  • 内部做了很多优化,包括从磁盘读取数据时采用的可预测性读、能够加快读操作并且自动创建的自适应哈希索引、能够加速插入操作的插入缓冲区等。

  • 支持真正的在线热备份。其它存储引擎不支持在线热备份,要获取一致性视图需要停止对所有表的写入,而在读写混合场景中,停止写入可能也意味着停止读取。

MyISAM

  • 设计简单,数据以紧密格式存储。对于只读数据,或者表比较小、可以容忍修复操作,则依然可以使用它。

  • 提供了大量的特性,包括压缩表、空间数据索引等。

  • 不支持事务。

  • 不支持行级锁,只能对整张表加锁,读取时会对需要读到的所有表加共享锁,写入时则对表加排它锁。但在表有读取操作的同时,也可以往表中插入新的记录,这被称为并发插入(CONCURRENT INSERT)。

总结

  • 事务: InnoDB 是事务型的,可以使用 Commit 和 Rollback 语句。

  • 并发: MyISAM 只支持表级锁,而 InnoDB 还支持行级锁。

  • 外键: InnoDB 支持外键。

  • 备份: InnoDB 支持在线热备份。

  • 崩溃恢复: MyISAM 崩溃后发生损坏的概率比 InnoDB 高很多,而且恢复的速度也更慢。

  • 其它特性: MyISAM 支持压缩表和空间数据索引。

14、数据库并发事务会带来哪些问题?

数据库并发会带来脏读、幻读、丢弃更改、不可重复读这四个常见问题,其中:

脏读:在第一个修改事务和读取事务进行的时候,读取事务读到的数据为 100,这是修改之后的数据,但是之后该事务满足一致性等特性而做了回滚操作,那么读取事务得到的结果就是脏数据了。

幻读:一般是 T1 在某个范围内进行修改操作(增加或者删除),而 T2 读取该范围导致读到的数据是修改之间的了,强调范围。

丢弃修改:两个写事务 T1 T2 同时对 A=0 进行递增操作,结果 T2 覆盖 T1,导致最终结果是 1 而不是 2,事务被覆盖

不可重复读:T2 读取一个数据,然后 T1 对该数据做了修改。如果 T2 再次读取这个数据,此时读取的结果和第一次读取的结果不同。

脏读

脏读

第一个事务首先读取 var 变量为 50,接着准备更新为 100 的时,并未提交,第二个事务已经读取 var 为 100,此时第一个事务做了回滚。最终第二个事务读取的 var 和数据库的 var 不一样。

幻读(幻影读)

幻读

T1 读取某个范围的数据,T2 在这个范围内插入新的数据,T1 再次读取这个范围的数据,此时读取的结果和和第一次读取的结果不同。

丢弃修改

丢弃修改

T1 和 T2 两个事务都对一个数据进行修改,T1 先修改,T2 随后修改,T2 的修改覆盖了 T1 的修改。例如:事务 1 读取某表中的数据 A=50,事务 2 也读取 A=50,事务 1 修改 A=A+50,事务 2 也修改 A=A+50,最终结果 A=100,事务 1 的修改被丢失。

不可重复读

不可重复读

T2 读取一个数据,T1 对该数据做了修改。如果 T2 再次读取这个数据,此时读取的结果和第一次读取的结果不同。

15、数据库悲观锁和乐观锁的原理和应用场景分别有什么?

悲观锁,先获取锁,再进行业务操作,一般就是利用类似 SELECT … FOR UPDATE 这样的语句,对数据加锁,避免其他事务意外修改数据。当数据库执行 SELECT … FOR UPDATE 时会获取被 select 中的数据行的行锁,select for update 获取的行锁会在当前事务结束时自动释放,因此必须在事务中使用。

乐观锁,先进行业务操作,只在最后实际更新数据时进行检查数据是否被更新过。Java 并发包中的 AtomicFieldUpdater 类似,也是利用 CAS 机制,并不会对数据加锁,而是通过对比数据的时间戳或者版本号,来实现乐观锁需要的版本判断。

16、MySQL 索引主要使用的两种数据结构是什么?

  • 哈希索引,对于哈希索引来说,底层的数据结构肯定是哈希表,因此在绝大多数需求为单条记录查询的时候,可以选择哈希索引,查询性能最快;其余大部分场景,建议选择 BTree 索引

  • BTree 索引,Mysql 的 BTree 索引使用的是 B 树中的 B+Tree,BTREE 索引就是一种将索引值按一定的算法,存入一个树形的数据结构中(二叉树),每次查询都是从树的入口 root 开始,依次遍历 node,获取 leaf。但对于主要的两种存储引擎(MyISAM 和 InnoDB)的实现方式是不同的。

17、数据库为什么要进行分库和分表呢?都放在一个库或者一张表中不可以吗?

18、不可重复读和幻读区别是什么?可以举个例子吗?

不可重复读的重点是修改,幻读的重点在于新增或者删除。

  • 例 1(同样的条件, 你读取过的数据, 再次读取出来发现值不一样了 ):事务 1 中的 A 先生读取自己的工资为 1000 的操作还没完成,事务 2 中的 B 先生就修改了 A 的工资为 2000,导致 A 再读自己的工资时工资变为 2000;这就是不可重复读。

  • 例 2(同样的条件, 第 1 次和第 2 次读出来的记录数不一样 ):假某工资单表中工资大于 3000 的有 4 人,事务 1 读取了所有工资大于 3000 的人,共查到 4 条记录,这时事务 2 又插入了一条工资大于 3000 的记录,事务 1 再次读取时查到的记 录就变为了 5 条,这样就导致了幻读。

19、MySQL 中有四种索引类型,可以简单说说吗?

  • FULLTEXT :即为全文索引,目前只有 MyISAM 引擎支持。其可以在 CREATE TABLE ,ALTER TABLE ,CREATE INDEX 使用,不过目前只有 CHAR、VARCHAR ,TEXT 列上可以创建全文索引。

  • HASH :由于 HASH 的唯一(几乎 100%的唯一)及类似键值对的形式,很适合作为索引。HASH 索引可以一次定位,不需要像树形索引那样逐层查找,因此具有极高的效率。但是,这种高效是有条件的,即只在“=”和“in”条件下高效,对于范围查询、排序及组合索引仍然效率不高。

  • BTREE :BTREE 索引就是一种将索引值按一定的算法,存入一个树形的数据结构中(二叉树),每次查询都是从树的入口 root 开始,依次遍历 node,获取 leaf。这是 MySQL 里默认和最常用的索引类型。

  • RTREE :RTREE 在 MySQL 很少使用,仅支持 geometry 数据类型,支持该类型的存储引擎只有 MyISAM、BDb、InnoDb、NDb、Archive 几种。相对于 BTREE,RTREE 的优势在于范围查找。

20、视图的作用是什么?可以更改吗?

视图是虚拟的表,与包含数据的表不一样,视图只包含使用时动态检索数据的查询;不包含任何列或数据。使用视图可以简化复杂的 sql 操作,隐藏具体的细节,保护数据;视图创建后,可以使用与表相同的方式利用它们。

视图不能被索引,也不能有关联的触发器或默认值,如果视图本身内有 order by 则对视图再次 order by 将被覆盖。

创建视图:create view xxx as xxxx

对于某些视图比如未使用联结子查询分组聚集函数 Distinct Union 等,是可以对其更新的,对视图的更新将对基表进行更新;但是视图主要用于简化检索,保护数据,并不用于更新,而且大部分视图都不可以更新。

21、为什么说 B+tree 比 B 树更适合实际应用中操作系统的文件索引和数据库索引?

B+tree 的磁盘读写代价更低,B+tree 的查询效率更加稳定 数据库索引采用 B+树而不是 B 树的主要原因:B+树只要遍历叶子节点就可以实现整棵树的遍历,而且在数据库中基于范围的查询是非常频繁的,而 B 树只能中序遍历所有节点,效率太低。

B+树的特点

  • 所有关键字都出现在叶子结点的链表中(稠密索引),且链表中的关键字恰好是有序的;

  • 不可能在非叶子结点命中;

  • 非叶子结点相当于是叶子结点的索引(稀疏索引),叶子结点相当于是存储(关键字)数据的数据层;

22、一道场景题:假如你所在的公司选择 MySQL 数据库作数据存储,一天五万条以上的增量,预计运维三年,你有哪些优化手段?

  • 设计良好的数据库结构,允许部分数据冗余,尽量避免 join 查询,提高效率。

  • 选择合适的表字段数据类型和存储引擎,适当的添加索引。

  • MySQL 库主从读写分离。

  • 找规律分表,减少单表中的数据量提高查询速度。

  • 添加缓存机制,比如 Memcached,Apc 等。

  • 不经常改动的页面,生成静态页面。

  • 书写高效率的 SQL。比如 SELECT * FROM TABEL 改为 SELECT field_1, field_2, field_3 FROM TABLE。

23、什么时候需要建立数据库索引呢?

在最频繁使用的、用以缩小查询范围的字段,需要排序的字段上建立索引。不宜:1)对于查询中很少涉及的列或者重复值比较多的列 2)对于一些特殊的数据类型,不宜建立索引,比如文本字段(text)等。

24、覆盖索引是什么?

如果一个索引包含(或者说覆盖)所有需要查询的字段的值,我们就称 之为“覆盖索引”。

我们知道在 InnoDB 存储引 擎中,如果不是主键索引,叶子节点存储的是主键+列值。最终还是要“回表”,也就是要通过主键再查找一次,这样就 会比较慢。覆盖索引就是把要查询出的列和索引是对应的,不做回表操作!

25、数据库中的主键、超键、候选键、外键是什么?(很棒)

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值