精华内容
下载资源
问答
  • 在业务代码已经保证插入身份证唯一的情况下,可以选择建立唯一索引普通索引,这时该如何选择呢?接下来,将从查询和更新的执行过程进行分析。 查询过程 假设 k 是表 t 上的索引,在搜索 select id from t where k=...
  • Mysql唯一索引普通索引的区别,

    千次阅读 2020-07-15 16:47:31
    文章目录Mysql唯一索引普通索引的区别,那种速度一点,原因是啥理由说明:结论:1 普通索引2 唯一索引注意:唯一索引普通索引使用的结构都是B-tree,执行时间复杂度都是O(log n)。3 主索引4 外键索引5 复合索引...

    Mysql唯一索引和普通索引的区别,那种速度快一点,原因是啥

    其实,如果业务上就要求我们数据库的值必须是唯一的,那没什么好讨论的,就选择唯一索引;那么如果业务上要求不严格,或者说不需要我们数据库后台来保障唯一性要求,这时候我们选择唯一索引还是普通索引就看谁在性能上更好,谁好选谁。

    唯一索引和普通索引怎么选择

    理由说明:

    我们都知道一个索引就是创建一个B+树,如果他是一个普通索引我们现在去查找,当查找到符合的数据的时候这个时候他就会在进行第二次查找,因为列里面的数据是唯一的,所以第二次查找肯定是没有数据的,这一次查找通过二分法去查找B+树查找到二叉树的叶子结点,每个叶子结点都会对应一个数据块(数据页),如果这个数据页在内存中那么他也就不用去在磁盘或硬盘去读区这个数据了,直接在内存中读取,我们也知道一个数据块大概是可以存储几千条数据的,所以在第一次查找的时候刚好这个数据块的尾部,在进行第二次查找的时候就要在进行IO读取了 一次IO读取是很低效的。但是如果是唯一索引的话这样的话就不会再进行第二次读取了,所以在最坏的情况下,普通索引和唯一索引的最大区别是多进行一次IO读取,概率极小。
    update:
    2.在update或者insert的时候普通索引也就是去在B+ 树中去找位置然后按照B+树的算法结构更新放入 或者更新其结构(拓展:在创建索引的时候会降低更新和插入效率,因为B+树在某种情况下添加结点效率会比较低),这在innoDB中如果说是在内存中没有这个数据块那么在这个情况下他不会去现在就去更新,会将这个更新操作存储在ChangeBuffer上,当第二次有查找这个数据的时候再去读取这个数据块然后再执行这个ChangeBuffer 中的数据操作,meger 这个数据。(在redlog里面也会存一份,也会存储changeBuffer的才操作数据,ES(Elaticsearch)的一种同步方式是通过redo log 来进行是实时同步的),当唯一索引进行更新操作的时候 会先进行一次查找是否唯一,再进行和普通索引一样的操作。所以在符合业务逻辑和无误的前提下:唯一索引比普通索引多一个查找操作。(在进行insert的时候会根据主键来进行查住 主键索引的数据结构发生改变)

    结论:

    在符合业务场景的前提下的时候唯一索引还是少用多用用普通索引。 还有就是change Buffer的使用 在更新读取少的 情况下放大change buffer的存储量 因为chanbuffer量越大其变更的数量更多 减少的IO次书更多,性能也就提高了。

    1 普通索引

    普通索引(由关键字KEY或INDEX定义的索引)的唯一任务是加快对数据的访问速度。因此,应该只为那些最经常出现在查询条件(WHEREcolumn=)或排序条件(ORDERBYcolumn)中的数据列创建索引。只要有可能,就应该选择一个数据最整齐、最紧凑的数据列(如一个整数类型的数据列)来创建索引。

    2 唯一索引

    • 普通索引允许被索引的数据列包含重复的值。比如说,因为人有可能同名,所以同一个姓名在同一个“员工个人资料”数据表里可能出现两次或更多次。
    • 如果能确定某个数据列将只包含彼此各不相同的值,在为这个数据列创建索引的时候就应该用关键字UNIQUE把它定义为一个唯一索引。
    • 这么做的好处:一是简化了mysql对这个索引的管理工作,这个索引也因此而变得更有效率;二是MySQL会在有新记录插入数据表时,自动检查新记录的这个字段的值是否已经在某个记录的这个字段里出现过了;如果是,MySQL将拒绝插入那条新记录。
    • 也就是说,唯一索引可以保证数据记录的唯一性。事实上,在许多场合,人们创建唯一索引的目的往往不是为了提高访问速度,而只是为了避免数据出现重复。

    注意:唯一索引和普通索引使用的结构都是B-tree,执行时间复杂度都是O(log n)。

    3 主索引

    在前面已经反复多次强调过:必须为主键字段创建一个索引,这个索引就是所谓的"主索引"。主索引与唯一索引的唯一区别是:前者在定义时使用的关键字是PRIMARY而不是UNIQUE。

    4 外键索引

    如果为某个外键字段定义了一个外键约束条件,MySQL就会定义一个内部索引来帮助自己以最有效率的方式去管理和使用外键约束条件。

    5 复合索引

    索引可以覆盖多个数据列,如像INDEX(columnA, columnB)索引。这种索引的特点是MySQL可以有选择地使用一个这样的索引。如果查询操作只需要用到columnA数据列上的一个索引,就可以使用复合索引INDEX(columnA, columnB)。不过,这种用法仅适用于在复合索引中排列在前的数据列组合。比如说,INDEX(A, B, C)可以当做A或(A, B)的索引来使用,但不能当做B、C或(B, C)的索引来使用。

    6 全文索引

    文本字段上的普通索引只能加快对出现在字段内容最前面的字符串(也就是字段内容开头的字符)进行检索操作。如果字段里存放的是由几个、甚至是多个单词构成的较大段文字,普通索引就没什么作用了。这种检索往往以LIKE %word%的形式出现,这对MySQL来说很复杂,如果需要处理的数据量很大,响应时间就会很长。
      这类场合正是全文索引(full-text index)可以大显身手的地方。在生成这种类型的索引时,MySQL将把在文本中出现的所有单词创建为一份清单,查询操作将根据这份清单去检索有关的数据记录。全文索引即可以随数据表一同创建,也可以等日后有必要时再使用下面这条命令添加:
      ALTER TABLE tablename ADD FULLTEXT(column1, column2)
      有了全文索引,就可以用SELECT查询命令去检索那些包含着一个或多个给定单词的数据记录了。下面是这类查询命令的基本语法:
      SELECT * FROM tablename
      WHERE MATCH(column1, column2) AGAINST(‘word1’, ‘word2’, ‘word3’)
      上面这条命令将把column1和column2字段里有word1、word2和word3的数据记录全部查询出来。
      注解:InnoDB数据表不支持全文索引。

    参考:
    唯一索引和普通索引怎么选择
    唯一索引和普通索引
    唯一索引和普通索引的区别

    展开全文
  • 我秃了!唯一索引普通索引我该选谁?

    千次阅读 多人点赞 2021-09-06 00:35:26
    你遇到过不知道如何选择唯一索引普通索引的场景么?你知道他俩的原理和区别么?来不及了,上车!!

    小伙伴想精准查找自己想看的MySQL文章?喏 → MySQL江湖路 | 专栏目录
    在这里插入图片描述

      提到唯一索引普通索引,相信大家都不陌生,当同事小姐姐问你这俩有什么区别时?或许你会脱口而出:“这还用问?见名知意啊,一个是允许字段重复,一个不允许存在重复数据!”

      是否解决小姐姐的疑问我不知道,但你在同事心目中,肯定不是啥好玩意儿~ 要知道,一眼就看出的答案,一般不会有人问,除非问傻子~

    那么当你处理一张市民信息表时,其中一列为市民的身份证号信息,你会怎么选择哪个索引?为什么?

      对于一个经历过风风雨雨、日日夜夜的程序员来说,需要你考虑的东西可不仅是重不重复这类问题,而是…

    在这里插入图片描述

    开个玩笑~~应当结合实际情况,对各个场景进行综合考虑。

      其实,如果在业务代码中保证了不会写入重复的身份证号,那么这两个选择逻辑上都是正确的。但是在SELECT和DML场景中,唯一索引和普通索引却有很多不同。

    1、在SELECT中,唯一索引和普通索引的区别

      本文测试引擎选择我们最常用的InnoDB,版本为MySQL8.0;

    假设,执行查询的语句是:

    select id from T where id_card = 666;
    

      (身份证太长,咱们用简单数据做演示)我们知道,MySQL的InnoDB采用的是B+树实现的索引结构,查找过程从B+树的树根起,按层搜索到666所在的叶子节点,然后取出该节点所在的数据页,把数据页读到内存后,通过二分法在数据页中定位id_card=666的行数据。

    在这里插入图片描述

    B+ 树的查找过程如上图:

    1. 将磁盘块1从磁盘加载到内存,发生一次IO ,在内存中使用二分查找方式找到 666 在600和700 之间,锁定磁盘块1的P2 指针。
    2. 通过磁盘块1 的 P2 指针地址把磁盘块3 加载到内存,发生第二次IO ,锁定磁盘块3 的 P2 指针
    3. 通过磁盘块3 的P2指针加载磁盘块7到内存,发生第三次 IO,同时根据二分查找找到666 查询结束。

    普通索引和唯一索引的定位方式:

    • 普通索引:查到第一条id_card=666 后,然后继续往后查找直到碰到第一个 id_card<>666 的记录时,结束。
    • 唯一索引:由于索引定义了唯一性,查找到第一个满足条件的记录后,直接结束。

      两者在查询方面的性能差距微乎其微。对于普通索引多的那一次操作,因为本身就是以数据页为单位读进内存,数据页大小默认16KB(大概1000行),要多做的那一次“查找和判断下一条记录”的操作,就只需要一次指针寻找和一次计算。当然,不可避免查询的数据是该数据页的最后一位,这样还要再读下一块数据页,算法会复杂一些。

      但你知道的,这种概率很小,我们程序员要相信逆墨菲定律:大概率不会出现且未被发现的BUG,在难以改动的前提下,你就当不知道就完了,发生了又能咋地?有测试顶着呢!

    在这里插入图片描述

      有同学问我了:普通索引为什么要继续向下查找?继续向下查找的原因是由于普通索引允许重复值,且B+Tree是天然有序的。SQL中并没有指定limit 1,所以他还要往下查,看是否有同条件的数据一起返回,直到查到第一条不满足条件的数据为止。

    2、在DML中,唯一索引和普通索引的区别

      ding!这是本篇文章的重点,在看之前,我们需要先了解什么是change buffer

      了解MySQL机制的同学们知道,当执行 DML(INSERT、UPDATE、DELETE)等操作时,InnoDB会利用 change buffer进行加速写操作,可以将写操作的随机磁盘访问调整为局部顺序操作,而在机械硬盘时代,随机磁盘访问(随机I/O)也是数据库操作中的最耗性能的硬伤。当普通索引(非唯一索引)的数据页发生写操作时,把操作内容写到内存中的change buffer后就可以立刻返回(执行完成)了。

      这里我以UPDATE操作为例,当需要更新某一行数据时,会先判断该行所在数据页是否在内存中,如果在就直接在内存数据页中更新,如果这个数据页没有内存中的话,在不影响数据一致性的前提下,InnoDB 会将这些UPDATE操作缓存在 change buffer 中,这样就不需要从磁盘读入数据页,当有SQL查询需要访问这个数据页的数据时,将数据页读入内存后,然后先执行 change buffer 中与这个页的相关UPDATE操作,通过这种方式保证这个数据页的逻辑正确性

    在这里插入图片描述

      可见,change buffer是会被从内存持久化到磁盘中的,将 change buffer 中的操作应用到原数据页,得到最新结果的过程被称为 merge。除了访问这个数据页会触发 merge 外,系统有后台线程会定期 merge。在数据库正常关闭(shutdown)的过程中,也会执行 merge 操作,相当于刷脏页啦(把已修改的数据更新到实际数据文件中)。

    触发merge的操作主要有以下几种(你该记住的点):

    • 有SQL线程访问这个数据页;
    • master thread线程每秒或每10秒进行一次merge change buffer的操作;
    • 在数据库正常关闭的时候。

      小朋友,你是否有很多问号??DB服务器宕机,数据不是就丢了?这就得redo log + binlog来保证了,可以参考作者另一篇文章《听我讲完redo log、binlog原理,面试官老脸一红》,本篇不再赘述。

      跑远了?言归正传~~上文提到普通索引(非唯一索引)会使用到change buffer进行加速写操作,你是不是已经get到点了呢~

      是的,唯一索引不会使用 Change buffer ,如果索引设置了唯一属性,在进行插入或者修改操作时,InnoDB 必须进行唯一性检查,如果不读取索引页到缓冲池,无法校验索引是否唯一,如果都把索引页读到内存了,那直接更新内存会更快,就没必要使用change buffer了。

    • 对于普通索引(非唯一索引)的DML操作来说,当待更新的数据页在内存中时,找到前值和后值的区间插入即可;当待更新的数据页在不在内存中时,直接把操作写到Change buffer就完事儿了。舒服!

    • 对于唯一索引,当待更新的数据页在不在内存中时,索引每次都得把数据页读到内存中判断唯一性,将数据从磁盘读入内存涉及大量随机IO的访问,慢的一批,当遇到高频写操作时??唉,别想了,难受!

      到这里,相信你对普通索引和唯一索引的取舍有了一定的概念,普通索引和唯一索引在查询能力上是没差别的,主要考虑的是更新的影响。还得结合实际业务场景来判断,如果是读取远大于更新和插入的表,唯一索引和普通索引都可以,但是如果业务需求相反,个人觉得应该使用普通索引,当然如果是那种更新完要求立即可见的需求,就是刚更新完就要再查询的,这种情况下反而不推荐普通索引,因为这样会频繁的产生merge操作,起不到change buffer的作用,反而需要额外空间来维护change buffer就有点得不偿失了。

      当我们使用普通索引,尤其在使用机械盘的场景下,尽量把change buffer开大从而确保数据的写入速度。最后,通过列举一下 change buffer 的配置,结束今天的分享,相信看到这里的都是有心人,也是喜爱MySQL的崽子,记得不要吝啬你的点赞哦~~

    在这里插入图片描述

    change buffer 配置

    • innodb_change_buffer_max_size% 配置写缓冲的大小,占整个缓冲池的比例,默认值是25%,可以通过修改该值提高InnoDB写效率,最大值是50%。
    mysql> show variables like '%innodb_change_buffer_max_size%';
    +-------------------------------+-------+
    | Variable_name                 | Value |
    +-------------------------------+-------+
    | innodb_change_buffer_max_size | 25    |
    +-------------------------------+-------+
    1 row in set (0.00 sec)
    
    • innodb_change_buffering配置是否缓存辅助索引页的修改,默认为 all,即缓存 INSERT/DELETE/UPDATE等DML操作。
    mysql> show variables like '%innodb_change_buffering%';
    +-------------------------+-------+
    | Variable_name           | Value |
    +-------------------------+-------+
    | innodb_change_buffering | all   |
    +-------------------------+-------+
    1 row in set (0.00 sec)
    

    MySQL系列文章汇总与《MySQL江湖路 | 专栏目录》

    往期热门MySQL系列文章

    展开全文
  • Mysql主键索引、唯一索引普通索引、全文索引、组合索引的区别

    Mysql索引概念:
    说说Mysql索引,看到一个很少比如:索引就好比一本书的目录,它会让你更快的找到内容,显然目录(索引)并不是越多越好,假如这本书1000页,有500也是目录,它当然效率低,目录是要占纸张的,而索引是要占磁盘空间的。


    Mysql索引主要有两种结构:B+树和hash.

    hash:hsah索引在mysql比较少用,他以把数据的索引以hash形式组织起来,因此当查找某一条记录的时候,速度非常快.当时因为是hash结构,每个键只对应一个值,而且是散列的方式分布.所以他并不支持范围查找和排序等功能.

    B+树:b+tree是mysql使用最频繁的一个索引数据结构,数据结构以平衡树的形式来组织,因为是树型结构,所以更适合用来处理排序,范围查找等功能.相对hash索引,B+树在查找单条记录的速度虽然比不上hash索引,但是因为更适合排序等操作,所以他更受用户的欢迎.毕竟不可能只对数据库进行单条记录的操作. 


    Mysql常见索引有:主键索引、唯一索引、普通索引、全文索引、组合索引

    PRIMARY KEY(主键索引)  ALTER TABLE `table_name` ADD PRIMARY KEY ( `column` ) UNIQUE(唯一索引)     ALTER TABLE `table_name` ADD UNIQUE (`column`)
    INDEX(普通索引)     ALTER TABLE `table_name` ADD INDEX index_name ( `column` ) FULLTEXT(全文索引)      ALTER TABLE `table_name` ADD FULLTEXT ( `column` )
    组合索引   ALTER TABLE `table_name` ADD INDEX index_name ( `column1`, `column2`, `column3` ) 

    Mysql各种索引区别:
    普通索引:最基本的索引,没有任何限制
    唯一索引:与"普通索引"类似,不同的就是:索引列的值必须唯一,但允许有空值。
    主键索引:它 是一种特殊的唯一索引,不允许有空值。 
    全文索引:仅可用于 MyISAM 表,针对较大的数据,生成全文索引很耗时好空间。
    组合索引:为了更多的提高mysql效率可建立组合索引,遵循”最左前缀“原则。

    展开全文
  • change buffer 唯一索引普通索引

    千次阅读 2020-01-03 16:55:51
    唯一索引普通索引 change buffer 学习检测 什么是change buffer? 唯一索引普通索引查找数据流程及性能对比? 唯一索引普通索引更新数据流程及性能对比? change buffer 适用场景? change buffer 和 ...

    唯一索引和普通索引 change buffer

    学习检测

    1. 什么是change buffer?

    2. 唯一索引和普通索引查找数据流程及性能对比?

    3. 唯一索引和普通索引更新数据流程及性能对比?

    4. change buffer 适用场景?

    5. change buffer 和 redo log 的区别?

    6. change buffer 是持久化的吗?

    7. chabge buffer 的merge 操作是什么?

    8. 数据搜索修改的流程,是否加载内存中?

    9. 当有where 条件的时候,change buffer 是否其作用,为什么?

    10. 二级索引中的普通索引也会更新主键索引,什么情况下change buffer起作用?

    11. change buffer 参数了解一下?默认值是多少

    12. change buffer 刷新数据到磁盘的情况?

    13. 服务掉电,change buffer 的数据会丢失吗?什么情况有丢失?

    14. 唯一索引可以change buffer 吗?

    总结

    1. change buffer 是对数据进行修改的时候(insert、update、delete)的时候将修改优先写入change buffer,减少磁盘的随机IO消耗

    2. 数据查找过程 以 select * from where id = 3;为例

      在内存中

      唯一索引,在索引树中找到id为3的就停止搜索,返回数据

      普通索引,在搜索到id为3的数据后继续向下搜索,直到找到第一个不满足条件的数据,返回

      不在内存中

      性能差异----微乎其微

      性能只是普通索引多一次判断,忽略不计。数据是以页加载到内存中处理的,除非次数据是数据页的最后一条,会再次进行一次磁盘访问

    3. 数据修改过程(无 where情况)

      在内存中

      唯一索引,修改数据,直接返回

      普通索引,修改数据,直接返回

      不在内存中

      唯一索引,加载数据所在页,判断唯一性,修改数据

      普通索引,修改数据到change buffer,返回

      普通和唯一索引的最大区别在于

      数据不在内存时候,唯一索引要加载数据判断唯一性,普通直接存入change buffer,返回

    4. change buffer 适用于普通索引和写多读少的场景,读的操作会出发数据merge操作

    5. change buffer 的内存变化页记录在redo log中,是减少磁盘的读IO,redo log是物理逻辑日志,记录的是数据页的状态,减少磁盘的写IO

    6. change buffer 除了在buffer pool中的内存拷贝,在系统表空间也有存储(ibdata1)中

    7. change buffer 的merge操作是将change buffer 中的数据操作记录刷新到磁盘,保证数据的一致性

    8. mysql 对数据的操作都是加载到内存中,以页的单位16k,此参数可以修改,扇区是512byte,一个block是4k

    9. 当有 where条件的操作,也是可以用到change buffer的,删除操作(当前索引标记为删除)更新动作是(插入和删除的结合)

    10. 缓存条件是

    在这里插入图片描述
    11. innodb_change_buffering值包括

    all  默认值:缓冲池插入、删除标记操作和清楚
    
    none 不做任何缓冲操作
    
    inserts 缓冲区插入操作
    
    deletes 缓冲区删除标记操作
    
    changes 缓冲插入和删除标记操作(update)
    
    purges 缓冲在后台发生的物理删除事件、
    
    innodb_change_buffer_max_size 占用缓冲池大小比例 默认25 最大50%
    
    1. 读取数据的时候

      后台线程定期刷新,默认10s

      服务shutdown的时候

    2. 服务掉电,数据记录在系统表空间和change buffer 中,这时候redo log中的事务为prepare的时候,还没有commit的操作,重启后,此事务会丢失

    3. 唯一索引,update 和insert的时候不可以,delete的时候可以

    官方文档 和 toabao mysql 月报 有说明(太牛逼了)

    https://dev.mysql.com/doc/refman/8.0/en/innodb-change-buffer.html

    http://mysql.taobao.org/monthly/2015/07/01/

    change buffer

    就是将二级索引的insert、update、delete的操作记录放在change buffer中,避免此类操作频繁进行磁盘IO

    change buffer会占用buffer pool的空间

    在磁盘上会占用系统表空间

    change buffer是存放二级索引的没有在buffer pool的变更页的缓存区,变更的buffer是由insert,update,delete等操作导致的。等页被加载进buffer pool中后会将change buffer中的页合并。

    当需要更新一个数据页时,如果数据页在内存中就直接更新,而如果这个数据页还没有在内存中的话,在不影响数据一致性的前提下,InooDB 会将这些更新操作缓存在 change buffer 中,这样就不需要从磁盘中读入这个数据页了。在下次查询需要访问这个数据页的时候,将数据页读入内存,然后执行 change buffer 中与这个页有关的操作。通过这种方式就能保证这个数据逻辑的正确性。

    需要说明的是,虽然名字叫作 change buffer,实际上它是可以持久化的数据。也就是说,change buffer 在内存中有拷贝,也会被写入到磁盘上。

    将 change buffer 中的操作应用到原数据页,得到最新结果的过程称为 merge。除了访问这个数据页会触发 merge 外,系统有后台线程会定期 merge。在数据库正常关闭(shutdown)的过程中,也会执行 merge 操作。

    显然,如果能够将更新操作先记录在 change buffer,减少读磁盘,语句的执行速度会得到明显的提升。而且,数据读入内存是需要占用 buffer pool 的,所以这种方式还能够避免占用内存,提高内存利用率。

    change buffer 触发merge条件

    • 访问这个数据页会触发merge操作,将change buffer操作应用到原始数据页

    • 后台线程会定期的merge操作,默认10s

    • 数据库正常关闭(shutdown)的过程中,也会执行merge操作

    change buffer 使用场景

    • change buffer 只限于用在普通索引的场景下,而不适用唯一索引

    • 对于写少读多的场景不适合,每次读操作都会触发merge操作,反而起到副作用

    • 适用于写多读少的场景

    对于唯一索引来说,**所有的更新操作都要先判断这个操作是否违反唯一性约束。比如,要插入 (4,400) 这个记录,就要先判断现在表中是否已经存在 k=4 的记录,而这必须要将数据页读入内存才能判断。**如果都已经读入到内存了,那直接更新内存会更快,就没必要使用 change buffer 了。

    因此,唯一索引的更新就不能使用 change buffer,实际上也只有普通索引可以使用。

    首先会进行merge操作后才能判断

    唯一索引不使用change buffer。唯一索引在执行更新操作时,要先判断是否符合唯一约束条件,那么在插入数据时会先从表中读取数据到内存中,然后判断是否符合唯一约束条件。所以唯一索引插入时肯定内存中会有这个数据页,即使没有也会查询磁盘数据保存到内存。所以内存中有数据页,直接更新即可,不需要保存到change buffer中。

    参数

    show variables like '%change_buffer%';
    

    在这里插入图片描述

    change buffer 和查询时候的数据怎么合并

    page hash 来记录change buffer 中是否有此条记录

    在这里插入图片描述

    查询区别

    查询语句

    select id from T where k=5
    

    这条查询语句在索引树上查找的过程,先是通过B+树丛树根开始,按层搜索到叶子节点,也就是图中右下角的这个数据页,然后通过数据也中的二分法来定位记录
    在这里插入图片描述

    • 对于普通索引来说,查找到满足条件的第一个记录 (5,500) 后,需要查找下一个记录,直到碰到第一个不满足 k=5 条件的记录。

    • 对于唯一索引来说,由于索引定义了唯一性,查找到第一个满足条件的记录后,就会停止继续检索。

    性能消耗的影响

    微乎其微

    因为数据树按照页的单位进行读写的,也就是说,当需要读一条记录的时候,并不是将这个记录本身从磁盘读出来,而是以页为单位,将其整体读入内存。在 InnoDB 中,每个数据页的大小默认是 16KB。

    因为引擎是按页读写的,所以说,当找到 k=5 的记录的时候,它所在的数据页就都在内存里了。那么,对于普通索引来说,要多做的那一次“查找和判断下一条记录”的操作,就只需要一次指针寻找和一次计算。

    当然,如果 k=5 这个记录刚好是这个数据页的最后一个记录,那么要取下一个记录,必须读取下一个数据页,这个操作会稍微复杂一些。

    但是,我们之前计算过,对于整型字段,一个数据页可以放近千个 key,因此出现这种情况的概率会很低。所以,我们计算平均性能差异时,仍可以认为这个操作成本对于现在的 CPU 来说可以忽略不计。

    修改性能区别

    如果要在这张表中插入一个新记录 (4,400) 的话,InnoDB 的处理流程是怎样的。

    数据在内存中

    • 对于唯一索引,找到3和5之间的位置,判断到有没有冲突,插入这个值.语句执行结束
    • 对于普通索引,找到3和5之间的位置,插入这个值,语句执行结束

    这样看来普通索引和唯一索引对更新语句性能的差别只是一个判断,置灰消耗微小的CPU时间

    数据页不在内存中

    • 对于唯一索引,需要将数据页读入内存,判断到没有冲突,插入这个值,语句执行结束;
    • 对于普通索引来说,则是将更新记录在 change buffer,语句执行就结束了

    将数据从磁盘读入内存涉及随机 IO 的访问,是数据库里面成本最高的操作之一。change buffer 因为减少了随机磁盘访问,所以对更新性能的提升是会很明显的。

    change buffer 和redo log

    mysql> insert into t(id,k) values(id1,k1),(id2,k2);
    

    情况1

    K1数据页子在内存中,k2数据也不再内存中
    在这里插入图片描述
    分析这条更新语句,你会发现它涉及了四个部分:内存、redo log(ib_log_fileX)、 数据表空间(t.ibd)、系统表空间(ibdata1)。

    1. page1在内存中,直接更新内存

    2. page2不在内存中,就在change buffer记录下”page2操作行的”信息

    3. 将这两个动作记录在redo log中,返回语句执行成功

    同时,图中的两个虚线箭头,是后台操作,不影响更新的响应时间。

    读请求

    select * from t where k in (k1, k2)。

    如果读语句发生在更新语句后不久,内存中的数据都还在,那么此时的这两个读操作就与系统表空间(ibdata1)和 redo log(ib_log_fileX)无关了。所以,我在图中就没画出这两部分。

    在这里插入图片描述

    1. 读page1的时候,直接从内存返回.有几位同学在前面文章的评论中问到,WAL 之后如果读数据,是不是一定要读盘,是不是一定要从 redo log 里面把数据更新以后才可以返回?其实是不用的。你可以看一下图 3 的这个状态,虽然磁盘上还是之前的数据,但是这里直接从内存返回结果,结果是正确的。

    2. 要读page2的时候,数据不在内存中,读取磁盘数据和change buffer 操作日志进行应用,生成一个新的正确版本,返回数据

    可以看到,直到需要读取page2的时候,这个数据页才被读取到内存中

    所以,如果要简单地对比这两个机制在提升更新性能上的收益的话,redo log 主要节省的是随机写磁盘的 IO 消耗(转成顺序写),而 change buffer 主要节省的则是随机读磁盘的 IO 消耗。

    展开全文
  • 昨天有位同事说,他的网页查询过程中发现普通索引唯一索引的效率是有差别的,普通索引比唯一索引快
  • 普通索引:最基本的索引,没有任何限制 唯一索引:与"普通索引"类似,不同的就是:索引列的值必须唯一,但允许有空值。 主键索引:它 是一种特殊...Mysql常见索引有:主键索引、唯一索引普通索引、全文...
  • 普通索引 这是最基本的索引类型,而且它没有唯一性之类的限制。 唯一性索引 这种索引和前面的“普通索引”基本相同,...对于唯一索引来说,由于索引定义了唯一性,查找到第一个满足条件的记录后,就会停止继续检索...
  • 点击链接 普通索引 唯一索引 全文索引 组合索引 ...
  • 一、MYSQL索引的分类 索引用于快速查找具有特定列值的行。如果没有索引,MySQL必须从第一行开始,然后读取整个表以查找...MYSQL的索引主要分为主键索引(PRIMARY KEY),唯一索引(UNIQUE),普通索引(INDEX)和全文索...
  • 一、背景介绍 索引用来快速地寻找那些具有特定值的记录,如果没有索引,执行查询时Mysql必须从第一个...如果表有一千个记录,通过索引查找记录至少要顺序扫描记录100倍。所以对于现在的各种大型数据库来说,索...
  • 对更新:唯一索引比普通索引更耗时. 查询流程:唯一索引找到第一个匹配数据后不继续往后查找. 1.普通索引,从索引树根节点开始按层往下查找,找到对应的数据页,然后二分法查找节点,找到第一个匹配的数据后,继续往后...
  • 如果在一个列上同时建唯一索引普通索引的话,mysql会自动选择唯一索引。 谷歌一下: 唯一索引普通索引使用的结构都是B-tree,执行时间复杂度都是O(log n)。 补充下概念:1、普通索引(非唯一索引) 普通索引(由...
  • 普通索引: 用来加速数据访问速度而建立的索引。多建立在经常出现在查询条件的字段和经常用于排序的字段。 被索引的数据列允许包含重复的值 唯一索引: 被索引的数据列不允许包含重复的值 以下内容转载自:...
  • MySQL 普通索引唯一索引和主索引

    千次阅读 2015-07-31 14:09:54
    1、普通索引  普通索引(由关键字KEY或INDEX定义的索引)的唯一任务是加快对数据的访问速度。因此,应该只为那些最经常出现在查询条件... 2、唯一索引  普通索引允许被索引的数据列包含重复的值。比如说,因为人
  • 主要给大家介绍了关于MySQL普通索引唯一索引的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
  • 问题:当一个字段同时满足唯一索引普通索引的情况下,我们要如何抉择呢? 要弄清楚这个问题,我们首先要了解唯一索引普通索引的实现原理,我们通过查询操作和更新操作分别来对比他们之间的区别 附带一提,普通...
  • 普通索引:不唯一也可为null;非聚合索引(可以查到记录对应的主键值,再使用主键的值通过索引找到需要的数据) 唯一索引:唯一可为null;唯一约束放在一 个或者多个列上,这些列或列的组合必须有唯一的;创建唯一...
  • 唯一索引普通索引性能比较

    千次阅读 2019-09-07 17:17:03
    1.普通索引 查找时,遇第一个不符合条件的才返回。 更新时,找到位置直接更新 2.唯一性索引 查找时,遇到第一个符合条件的直接返回。 更新时,先查找是否唯一,再找到位置然后再更新。 我们来通过底层分析一下: ...
  • 普通索引唯一索引

    万次阅读 多人点赞 2019-04-08 20:20:06
    所谓普通索引,就是在创建...普通索引还是唯一索引? 假设你在维护一个市民系统,每个人都有一个唯一的身份证号,而且业务代码已经保证了不会写入两个重复的身份证号。如果市民系统需要按照身份证号查姓名,就会...
  • 假设一个用户管理系统,每个人注册都有一个唯一的手机号,而且...2. 创建一个普通索引 如果业务代码已经保证了不会写入重复的身份证号,那么这两个选择逻辑上都是正确的。 从性能的角度考虑,选择唯一索引还是普通
  • mysql唯一索引普通索引的选择

    千次阅读 2019-02-27 14:19:27
    如果业务能够保证唯一性应该尽量选择普通索引而不是唯一索引 从查询和更新两方面说明: 查询: 查询来说二者的区别是 普通索引查到符合条件的项后会继续查找下一项,如果下一项不符合再返回 唯一索引则是查到符合...
  • Mysql索引介绍及常见索引(主键索引、唯一索引普通索引、全文索引、组合索引)的区别Mysql索引概念: 说说Mysql索引,看到一个很少比如:索引就好比一本书的目录,它会让你更的找到内容,显然目录(索引)并不是越...
  • Mysql主键索引、唯一索引普通索引、全文索引、组合索引的区别 MySQL索引原理及慢查询优化 Mysql索引概念: 说说Mysql索引,看到一个很少比如:索引就好比一本书的目录,它会让你更的找到内容,...
  • 对原文章排版进行了一些改进,方便查看,以下为...1、普通索引 (1)定义:最基本的索引类型,没有唯一性之类的限制。 (2)创建方式: a、创建索引,例如CREATE INDEX ON tablename (列的列表); b、修改表,例如ALTER
  • 1、普通索引  普通索引(由关键字KEY或INDEX定义的索引)的唯一任务是加快对数据的访问速度。因此,应该只为那些最经常出现在查询条件(WHERE... 2、唯一索引  普通索引允许被索引的数据列包含重复的值。

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 137,487
精华内容 54,994
关键字:

唯一索引比普通索引快