此错误是执行到分析器阶段报出的,由于 MySQL 会在分析器阶段检查 SQL 语句的正确性。面试
MySQL 查询缓存功能是在链接器以后发生的,它的优势是效率高,若是已经有缓存则会直接返回结果。 查询缓存的缺点是失效太频繁致使缓存命中率比较低,任何更新表操做都会清空查询缓存,所以致使查询缓存很是容易失效。正则表达式
MySQL 查询缓存默认是开启的,配置 querycachetype 参数为 DEMAND(按需使用)关闭查询缓存,MySQL 8.0 以后直接删除了查询缓存的功能。redis
MySQL 的经常使用引擎有 InnoDB、MyISAM、Memory 等,从 MySQL 5.5.5 版本开始 InnoDB 就成为了默认的存储引擎。sql
能够针对不一样的表设置不一样的引擎。在 create table 语句中使用 engine=引擎名(好比Memory)来设置此表的存储引擎。完整代码以下:数据库
create table student( id int primary key auto_increment, username varchar(120), age int ) ENGINE=Memory
InnoDB 和 MyISAM 最大的区别是 InnoDB 支持事务,而 MyISAM 不支持事务,它们主要区别以下:缓存
1)插入缓冲(insert buffer):对于非汇集索引的插入和更新,不是每一次直接插入索引页中,而是首先判断插入的非汇集索引页是否在缓冲池中,若是在,则直接插入,不然,先放入一个插入缓冲区中。好似欺骗数据库这个非汇集的索引已经插入到叶子节点了,而后再以必定的频率执行插入缓冲和非汇集索引页子节点的合并操做,这时一般能将多个插入合并到一个操做中,这就大大提升了对非汇集索引执行插入和修改操做的性能。安全
2)两次写(double write):两次写给 InnoDB 带来的是可靠性,主要用来解决部分写失败(partial page write)。doublewrite 有两部分组成,一部分是内存中的 doublewrite buffer ,大小为 2M,另一部分就是物理磁盘上的共享表空间中连续的 128 个页,即两个区,大小一样为 2M。当缓冲池的做业刷新时,并不直接写硬盘,而是经过 memcpy 函数将脏页先拷贝到内存中的 doublewrite buffer,以后经过 doublewrite buffer 再分两次写,每次写入 1M 到共享表空间的物理磁盘上,而后立刻调用 fsync 函数,同步磁盘。以下图所示服务器
3)自适应哈希索引(adaptive hash index):因为 InnoDB 不支持 hash 索引,但在某些状况下 hash 索引的效率很高,因而出现了 adaptive hash index 功能, InnoDB 存储引擎会监控对表上索引的查找,若是观察到创建 hash 索引能够提升性能的时候,则自动创建 hash 索引。并发
若是这张表的引擎是 MyISAM,那么 ID=4,若是是 InnoDB 那么 ID=2(MySQL 8 以前的版本)。函数
如下状况会致使 MySQL 自增主键不能连续:
自增主键能不能被持久化,说的是 MySQL 重启以后 InnoDB 能不能恢复重启以前的自增列,InnoDB 在 8.0 以前是没有持久化能力的,但 MySQL 8.0 以后就把自增主键保存到 redo log(一种日志类型,下文会详细讲)中,当 MySQL 重启以后就会从 redo log 日志中恢复。
共享表空间:指的是数据库的全部的表数据,索引文件所有放在一个文件中,默认这个共享表空间的文件路径在 data 目录下。 独立表空间:每个表都将会生成以独立的文件方式来进行存储。 共享表空间和独立表空间最大的区别是若是把表放再共享表空间,即便表删除了空间也不会删除,因此表依然很大,而独立表空间若是删除表就会清除空间。
独立表空间是由参数 innodbfileper_table 控制的,把它设置成 ON 就是独立表空间了,从 MySQL 5.6.6 版本以后,这个值就默认是 ON 了。
使用重建表的方式能够收缩表空间,重建表有如下三种方式:
表结构定义占有的存储空间比较小,在 MySQL 8 以前,表结构的定义信息存在以 .frm 为后缀的文件里,在 MySQL 8 以后,则容许把表结构的定义信息存在系统数据表之中。
覆盖索引是指,索引上的信息足够知足查询请求,不须要再回到主键上去取数据。
能够回表查询,若是把主键删掉了,那么 InnoDB 会本身生成一个长度为 6 字节的 rowid 做为主键。
多是由于 update 语句执行完成后,InnoDB 只保证写完了 redo log、内存,可能还没来得及将数据写到磁盘。
脏读是一个事务在处理过程当中读取了另一个事务未提交的数据;幻读是指同一个事务内屡次查询返回的结果集不同(好比增长了或者减小了行记录)。
由于行锁只能锁定存在的行,针对新插入的操做没有限定,因此就有可能产生幻读。 幻读带来的问题以下:
使用间隙锁的方式来避免出现幻读。间隙锁,是专门用于解决幻读这种问题的锁,它锁的了行与行之间的间隙,可以阻塞新插入的操做 间隙锁的引入也带来了一些新的问题,好比:下降并发度,可能致使死锁。
在 MySQL 的命令行中使用 show processlist;
查看全部链接,其中 Command 列显示为 Sleep 的表示空闲链接,以下图所示:
MySQL 的字符串类型和取值以下:
类型 | 取值范围 |
---|---|
CHAR(N) | 0~255 |
VARCHAR(N) | 0~65536 |
TINYBLOB | 0~255 |
BLOB | 0~65535 |
MEDUIMBLOB | 0~167772150 |
LONGBLOB | 0~4294967295 |
TINYTEXT | 0~255 |
TEXT | 0~65535 |
MEDIUMTEXT | 0~167772150 |
LONGTEXT | 0~4294967295 |
VARBINARY(N) | 0~N个字节的变长字节字符集 |
BINARY(N) | 0~N个字节的定长字节字符集 |
VARCHAR 和 CHAR 最大区别就是,VARCHAR 的长度是可变的,而 CHAR 是固定长度,CHAR 的取值范围为1-255,所以 VARCHAR 可能会形成存储碎片。因为它们的特性决定了 CHAR 比较适合长度较短的字段和固定长度的字段,如身份证号、手机号等,反之则适合使用 VARCHAR。
MySQL 存储金额应该使用 decimal
,由于若是存储其余数据类型,好比float
有致使小数点后数据丢失的风险。
去除前三条数据以后查询两条信息。
now() 返回当前时间包含日期和时分秒,current_date() 只返回当前时间,以下图所示:
使用 distinct 去重,使用 count 统计总条数,具体实现脚本以下:
select count(distinct f) from t
lastinsertid() 用于查询最后一次自增表的编号,它的特色是查询时不须要不须要指定表名,使用 select last_insert_id()
便可查询,由于不须要指定表名因此它始终以最后一条自增编号为主,能够被其它表的自增编号覆盖。好比 A 表的最大编号是 10,lastinsertid() 查询出来的值为 10,这时 B 表插入了一条数据,它的最大编号为 3,这个时候使用 lastinsertid() 查询的值就是 3。
删除数据有两种方式:delete 和 truncate,它们的区别以下:
delete 和 truncate 的使用脚本以下:
delete from t where username='redis'; truncate table t;
MySQL 中支持两种模糊查询:regexp 和 like,like 是对任意多字符匹配或任意单字符进行模糊匹配,而 regexp 则支持正则表达式的匹配方式,提供比 like 更多的匹配方式。 regexp 和 like 的使用示例以下: select * from person where uname like '%SQL%';> select from person where uname regexp '.SQL*.';
MySQL 支持枚举,它的实现方式以下:
create table t( sex enum('boy','grid') default 'unknown' );
枚举的做用是预约义结果值,当插入数据不在枚举值范围内,则插入失败,提示错误 Data truncated for column 'xxx' at row n
。
count(column) 和 count() 最大区别是统计结果可能不一致,count(column) 统计不会统计列值为 null 的数据,而 count() 则会统计全部信息,因此最终的统计结果可能会不一样。
A. count 的查询性能在各类存储引擎下的性能都是同样的。 B. count 在 MyISAM 比 InnoDB 的性能要低。 C. count 在 InnoDB 中是一行一行读取,而后累计计数的。 D. count 在 InnoDB 中存储了总条数,查询的时候直接取出。
答:C
由于 InnoDB 使用了事务实现,而事务的设计使用了多版本并发控制,即便是在同一时间进行查询,获得的结果也可能不相同,因此 InnoDB 不能把结果直接保存下来,由于这样是不许确的。
不能,由于 show table status 是经过采样统计估算出来的,官方文档说偏差可能在 40% 左右,因此 show table status 中的表行数不能直接使用。
A. select count(*) from t where time>1000 and time<4500 B. show table status where name='t' C. select count(id) from t where time>1000 and time<4500 D. select count(name) from t where time>1000 and time<4500
答:B 题目解析:由于 show table status 的表行数是估算出来,而其余的查询由于添加了 where 条件,即便是 MyISAM 引擎也不能直接使用已经存储的总条数,因此 show table status 的查询性能最高。
MyISAM 效率最高,由于 MyISAM 内部维护了一个计数器,直接返回总条数,而 InnoDB 要逐行统计。
MySQL 有对 count() 进行优化,以 InnoDB 为例,在 InnoDB 中主键索引数的叶子节点是主键值,而普通索引的叶子节点则是主键值,因此普通索引数比主键索引数要小不少,而对于 count 查询来讲,每一个索引树的查询结果都是同样的,因此 MySQL 会选择最小的那颗树来遍历,以此来优化 count() 的查询。
count(字段)<count(主键 id)<count(1)≈count(*) 题目解析:
因此最后得出的结果是:count(字段)<count(主键 id)<count(1)≈count(*)。
视图是一种虚拟的表,具备和物理表相同的功能,能够对视图进行增、改、查操做。视图一般是一个表或者多个表的行或列的子集。 视图建立脚本以下:
create view vname as select column_names from table_name where condition
MySQL 中的“视图”概念有两个,它们分别是:
能够用 Flashback 工具经过闪回把数据恢复回来。
Flashback 恢复数据的原理是是修改 binlog 的内容,拿回原库重放,从而实现数据找回。