我的编程空间,编程开发者的网络收藏夹
学习永远不晚

MySQL如何给字符串字段加索引

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

MySQL如何给字符串字段加索引

这篇文章给大家分享的是有关MySQL如何给字符串字段加索引的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

假设,你现在维护一个支持邮箱登录的系统,用户表是这么定义的:

create table SUser( ID bigint unsigned primary key, email varchar(64),  ...  )engine=innodb;

由于要使用邮箱登录,所以业务代码中一定会出现类似于这样的语句:

select f1, f2 from SUser where email='xxx';

如果 email 这个字段上没有索引,那么这个语句就只能做全表扫描。

1)那我可以在邮箱地址这个字段上面建索引吗?

  • MySQL 是支持前缀索引的,可以定义字符串的一部分作为索引

2)如果创建索引的语句不指定前缀长度,那么会怎么样?

  • 索引就会包含整个字符串

3)能举例来说明一下吗?

alter table SUser add index index1(email); 或 alter table SUser add index index2(email(6));
  • index1 索引里面,包含了每个记录的整个字符串

  • index2 索引里面,对于每个记录都是只取前 6 个字节

4)这两种不同的定义在数据结构和存储上有什么区别呢?

MySQL如何给字符串字段加索引

MySQL如何给字符串字段加索引

明显看出email(6) 这个索引结构占用的空间会更小

5)email(6) 这个索引结构有什么缺点吗?

  • 可能会增加额外的记录扫描次数

6)下面这个语句,在这两个索引定义下分别是怎么执行的?

select id,name,email from SUser where email='zhangssxyz@xxx.com';

index1(即 email 整个字符串的索引结构),执行顺序

  • 从 index1 索引树找到满足索引值是’zhangssxyz@xxx.com’的这条记录,取得 ID2 的值;

  • 回表查到主键值是 ID2 的行,判断 email 的值是正确的,将这行记录加入结果集;

  • 继续在index索引树的下一条记录,发现已经不满足 email='zhangssxyz@xxx.com’的条件了,循环结束。

这个过程中,只需要回主键索引取一次数据,所以系统认为只扫描了一行。

index2(即 email(6) 索引结构),执行顺序

  • 从 index2 索引树找到满足索引值是’zhangs’的记录,找到的第一个是 ID1;

  • 到主键上查到主键值是 ID1 的行,判断出 email 的值不是’zhangssxyz@xxx.com’,这行记录丢弃;

  • 取 index2 上刚刚查到的位置的下一条记录,发现仍然是’zhangs’,取出 ID2,再到 ID 索引上取整行然后判断,这次值对了,将这行记录加入结果集;

  • 重复上一步,直到在 idxe2 上取到的值不是’zhangs’时,循环结束。

在这个过程中,要回主键索引取 4 次数据,也就是扫描了 4 行。

7)通过上面的对比,能得出什么结论?

  • 使用前缀索引后,可能会导致查询语句读数据的次数变多。

8)前缀索引真的一无是处吗?

  • 如果我们定义的 index2 不是 email(6) 而是 email(7),那满足前缀’zhangss’的记录只有一个,直接就查到 ID2了,只扫描一行就结束了。

9)那么使用前缀索引有哪些注意事项?

  • 长度选择合理

10)当要给字符串创建前缀索引时,我咋知道我该用多长的前缀索引呢?

  • 统计索引上有多少个不同的值来判断要使用多长的前缀。

11)怎样统计索引上有多少个不同的值?

select count(distinct email) as L from SUser;

12)拿到了索引对应的有多少个不同的值之后下一步该做什么?

  • 依次选取不同长度的前缀来看这个值

    select    count(distinct left(email,4))as L4,   count(distinct left(email,5))as L5,   count(distinct left(email,6))as L6,   count(distinct left(email,7))as L7, from SUser;
  • 然后,在 L4~L7 中,找出第一个不小于 L * 95% 的值,说明通过这个索引可以找出百分之95以上的数据。

13)前缀索引对覆盖索引的影响是什么?

下面这个 SQL 语句:

select id,email from SUser where email='zhangssxyz@xxx.com';

与前面例子中的 SQL 语句

select id,name,email from SUser where email='zhangssxyz@xxx.com';

相比,第一个语句只要求返回 id 和 email 字段。

  • 如果使用 index1(即 email 整个字符串的索引结构)的话,查email的话就能得到ID,那就不用回表了,这个就是覆盖索引。

  • 用 index2(即 email(6) 索引结构)的话,就不得不回到 ID 索引再去判断 email 字段的值。

14)那我把index2 的定义修改为 email(18) 的前缀索引不就行了?

  • 这个18是你自己定义的,系统不知道18这个长度是否已经大于我的email长度,所以它还是会回表去查一下验证。

总而言之:使用前缀索引就用不上覆盖索引对查询性能的优化了

15)对于类似于邮箱这样的字段来说,使用前缀索引的效果可能还不错。但是,遇到身份证这种前缀的区分度不够好的情况时,我们要怎么办呢?

  • 索引选取的要更长一些。

    • 但是所以越长的话,占的磁盘空间更大,相同的一页能放下的索引值就变少了,反而会影响查询效率。

16)如果我们能够确定业务需求里面只有按照身份证进行等值查询的需求,还有没有别的处理方法呢?

  • 既然正过来相同的多,那我就把它倒过来存。查询时候这样查

    select field_list from t where id_card = reverse('input_id_card_string');

    使用 的时候用count(distinct) 方法去做个验证

  • 使用 hash 字段。在表上再创建一个整数字段,来保存身份证的校验码,同时在这个字段上创建索引。

    alter table t add id_card_crc int unsigned, add index(id_card_crc);

    每次插入新记录的时候,都同时用 crc32() 这个函数得到校验码填到这个新字段。由于校验码可能存在冲突,也就是说两个不同的身份证号通过 crc32() 函数得到的结果可能是相同的,所以你的查询语句 where 部分要判断 id_card 的值是否精确相同。

    select field_list from t where id_card_crc=crc32('input_id_card_string') and id_card='input_id_card_string'

    这样,索引的长度变成了 4 个字节(int类型),比原来小了很多

17)使用倒序存储和使用 hash 字段这两种方法有什么异同点?

  • 相同点:都不支持范围查询

    • 倒序存储的字段上创建的索引是按照倒序字符串的方式排序的,已经没有办法利用索引方式查出身份证号码在[ID_X, ID_Y]的所有市民了。同样地,hash 字段的方式也只能支持等值查询。

  • 区别

    • 从占用的额外空间来看,倒序存储方式在主键索引上,不会消耗额外的存储空间,而 hash 字段方法需要增加一个字段。当然,倒序存储方式使用 4 个字节的前缀长度应该是不够的,如果再长一点,这个消耗跟额外这个 hash 字段也差不多抵消了。

    • 在 CPU 消耗方面,倒序方式每次写和读的时候,都需要额外调用一次 reverse 函数,而 hash 字段的方式需要额外调用一次 crc32() 函数。如果只从这两个函数的计算复杂度来看的话,reverse 函数额外消耗的 CPU 资源会更小些。

    • 从查询效率上看,使用 hash 字段方式的查询性能相对更稳定一些。因为 crc32 算出来的值虽然有冲突的概率,但是概率非常小,可以认为每次查询的平均扫描行数接近 1。而倒序存储方式毕竟还是用的前缀索引的方式,也就是说还是会增加扫描行数。

案例:如果你在维护一个学校的学生信息数据库,学生登录名的统一格式是”学号 @gmail.com", 而学号的规则是:十五位的数字,其中前三位是所在城市编号、第四到第六位是学校编号、第七位到第十位是入学年份、最后五位是顺序编号。

18)系统登录的时候都需要学生输入登录名和密码,验证正确后才能继续使用系统。就只考虑登录验证这个行为的话,你会怎么设计这个登录名的索引呢?

  • 一个学校每年预估2万新生,50年才100万记录,能节省多少空间,直接全字段索引。省去了开发转换及局限性风险,碰到超大量迫不得已再用后两种办法

  • 实际操作上直接全字段索引就行了,一个学校数据库的数据量和查询压力都不会大到哪儿去。 如果单从优化数据表的角度: \1. 后缀@gmail可以单独一个字段来存,或者用业务代码来保证, \2. 城市编号和学校编号估计也不会变,也可以用业务代码来配置 \3. 然后直接存年份和顺序编号就行了,这个字段可以全字段索引

感谢各位的阅读!关于“MySQL如何给字符串字段加索引”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

MySQL如何给字符串字段加索引

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

MySQL如何给字符串字段加索引

这篇文章给大家分享的是有关MySQL如何给字符串字段加索引的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。假设,你现在维护一个支持邮箱登录的系统,用户表是这么定义的:create table SUser( ID b
2023-06-29

Mysql性能优化:如何给字符串加索引?

导读现代大部分的登录系统都支持邮箱、手机号码登录两种方式,那么如何在邮箱或者手机号码这个字符串上建立索引才能保证性能最佳呢?今天这篇文章就来探讨一下在Mysql中如何给一个字符串加索引才能达到性能最佳。本文首发于作者的微信公众号【码猿技术专栏】,原创不易,喜欢
Mysql性能优化:如何给字符串加索引?
2018-08-13

MySQL怎么给字符串加一个高效索引

这篇文章主要介绍了MySQL怎么给字符串加一个高效索引的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇MySQL怎么给字符串加一个高效索引文章都会有所收获,下面我们一起来看看吧。需求在日常需求中,用户使用手机号或
2023-07-05

MySQL给字符串加一个高效索引的实现

目录需求前缀索引倒序+前缀索引总结需求在日常需求中,用户使用手机号或者邮箱登录某一个系统,是一个很常见的操作,那如何在类似手机号或者邮箱这样的字段上建立一个合理的索引呢?前缀索引前缀索引,就是以一个字段值的一部分作为索引。我们在mys
2023-03-20

如何在 MySQL 中给表添加字段?(mysql怎么给表添加字段)

在MySQL数据库中,给表添加字段是一项常见的操作,以下是详细的步骤:一、准备工作确保你已经拥有MySQL数据库的访问权限,并且知道要操作的数据库名称和表名称。你可以使用MySQL客户端工具(如MySQLWork
如何在 MySQL 中给表添加字段?(mysql怎么给表添加字段)
2024-12-13

oracle怎么给表字段添加索引

要给表字段添加索引,可以使用Oracle的CREATE INDEX语句。语法如下:```CREATE INDEX index_nameON table_name (column_name);```其中,index_name是要创建的索引的名
2023-08-28

超长字符串字段,前缀索引两宗罪

前缀索引并不是一个难理解的东西,但是这里面涉及到的一些细节,我相信很多同学都没有去深入了解过。

MySQL字符串前缀索引使用

本文主要介绍了MySQL字符串前缀索引使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-02-03

mysql如何给表字段添加注释

在MySQL中,可以使用ALTER TABLE语句给表字段添加注释。具体操作如下:打开MySQL命令行工具或者使用可视化工具连接到MySQL数据库。选择数据库:使用USE语句选择要添加注释的表所在的数据库。例如,如果要添加注释的表位于名为"
2023-10-24

mysql如何给表增加一个字段

要给MySQL数据库中的表增加一个字段,可以使用ALTER TABLE语句。语法如下:ALTER TABLE 表名 ADD 列名 数据类型 [约束条件];例子:ALTER TABLE users ADD email VARCHAR(
mysql如何给表增加一个字段
2024-04-09

如何在mysql字符串中提取相应字段

从 mysql 字符串中提取特定字段:确定目标字段。使用正则表达式 regexp_extract() 匹配字段。使用子查询提取字段。使用字符串函数 substring() 和 position() 提取字段。如何从 MySQL 字符串中提取
如何在mysql字符串中提取相应字段
2024-06-15

MySQL添加索引及添加字段并建立索引方式

本指南提供有关MySQL索引的全面说明,包括添加索引的方法和建立索引的最佳实践。了解何时添加索引,例如处理大量数据和频繁的搜索和排序查询。此外,本文还介绍了添加字段并建立索引以避免性能问题,以及索引维护和优化技巧,例如使用批量插入和重建大型表。
MySQL添加索引及添加字段并建立索引方式
2024-04-02

java字符串下标索引怎么添加

在Java中,字符串是不可变的,即不能直接修改字符串的内容。但你可以通过连接操作符(+)或使用StringBuilder类来创建一个新的字符串对象。使用连接操作符(+):String str = "Hello";str = str +
java字符串下标索引怎么添加
2024-02-29

MySQL如何给表和字段添加注释

本文介绍了在MySQL中给表和字段添加注释的方法,包括语法、示例和查看注释的命令。注释可以提高数据库的可读性、文档化数据结构、简化代码维护并支持自动文档生成。最佳实践包括使用简洁的语句、包含详细的字段信息,并保持注释的最新状态。
MySQL如何给表和字段添加注释
2024-04-02

MySQL如何给表和字段添加注释

目录mysql给表和字段添加注释1.添加表和字段的注释2.修改表注释3.修改字段注释4.查询字段信息mysql添加新字段和注释,添加表注释总结mysql给表和字段添加注释在MySQL中,给表和字段添加注释使用COMMENT关键字。1.添
MySQL如何给表和字段添加注释
2024-02-01

编程热搜

目录