Mysql 的InnoDB引擎下支持hash索引吗?
什么,这个不是很明显支持的吗?还总是被面试官问到hash索引的btree索引有什么区别?
那么真的支持吗?
场景
很多人可能在使用Navicat给表创建索引时会发现,索引方法中支持BTREE和HASH
乍一看,很多人的第一反应是,这不是支持hash索引吗?
实践
那么我们来实践一下。
CREATE TABLE `auth_user` ( `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', `username` varchar(20) DEFAULT NULL COMMENT '用户名', `password` varchar(20) DEFAULT NULL COMMENT '密码', PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
对应sql语句:
ALTER TABLE oauth.auth_user ADD INDEX index1(username) USING HASH COMMENT '测试hash索引';
保存成功,表的DML语句变为:
CREATE TABLE `auth_user` ( `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', `username` varchar(20) DEFAULT NULL COMMENT '用户名', `password` varchar(20) DEFAULT NULL COMMENT '密码', PRIMARY KEY (`id`), KEY `index1` (`username`) USING HASH COMMENT '测试hash索引' ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
很多人可能会立刻说,这不是创建成功了吗?
我们继续,重新打开数据库连接,打开表设计,会发现index1索引的索引方法变成了btree,
但是表定义语句还是没有变化,
CREATE TABLE `auth_user` ( `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', `username` varchar(20) DEFAULT NULL COMMENT '用户名', `password` varchar(20) DEFAULT NULL COMMENT '密码', PRIMARY KEY (`id`), KEY `index1` (`username`) USING HASH COMMENT '测试hash索引' ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
那么这时,索引index1到底是btree索引还是hash索引呢?
通过命令查看表auth上的索引信息
show index from auth_user;
发现主键索引和index1索引的index_type都是btree,说明创建的hash索引是没有生效的。
分析:
查一下mysql官方文档:https://dev.mysql.com/doc/refman/5.7/en/create-index.html,
从上面的图中可以得知,InnoDB和MyISAM只支持BTree索引。
我们具体查一下InnoDB文档那一部分:https://dev.mysql.com/doc/refman/5.7/en/innodb-introduction.html。
可以发现InnoDB支持的所有特性,其中对Hash index特征的支持描述是:
No (InnoDB utilizes hash indexes internally for its Adaptive Hash Index feature.)
翻译过来:
不支持(InnoDB在内部利用hash索引来实现其自适应hash索引特性)
那么什么是Adaptive Hash Index呢?
根据文件,不难发现,自适应索引是InnoDB引擎的内存结构中的一种特性。
对自适应hash索引的描述:
自适应hash索引特性使InnoDB能够在具有适当的工作负载和足够的缓冲池内存的系统上执行更像内存中的数据库,而不牺牲事务特性或可靠性。
总的来说就是提高了查询性能。
那么怎样启动自适应hash索引的特性呢?
14.15 InnoDB Startup Options and System Variables System variables that are true or false can be enabled at server startup by naming them, or disabled by using a --skip- prefix. For example, to enable or disable the InnoDB adaptive hash index, you can use --innodb-adaptive-hash-index or --skip-innodb-adaptive-hash-index on the command line, or innodb_adaptive_hash_index or skip_innodb_adaptive_hash_index in an option file.
翻译过来就是可以在启动命令中,加上--innodb-adaptive-hash-index就可以开启InnoDB 自适应索引的特性了。
其实【MySQL技术内幕InnoDB存储引擎.姜承尧.扫描版】也提到了这点
总结:
Mysql InnoDB引擎不支持hash索引,但是在内存结构中有一个自适应hash索引,来提高查询性能。