首页 > 数据库 >18-mysql索引

18-mysql索引

时间:2023-12-27 15:26:34浏览次数:42  
标签:index name 18 索引 mysql where id select

一、b+树

image

每次查找数据时把磁盘IO次数控制在一个很小的数量级,最好是常数数量级。那么我们就想到如果一个高度可控的多路搜索树是否能满足需求呢?就这样,b+树应运而生(B+树是通过二叉查找树,再由平衡二叉树,B树演化而来)。

1.索引字段要尽量的小:通过上面的分析,我们知道IO次数取决于b+数的高度h,假设当前数据表的数据为N,每个磁盘块的数据项的数量是m,则有h=㏒(m+1)N,当数据量N一定的情况下,m越大,h越小;而m = 磁盘块的大小 / 数据项的大小,磁盘块的大小也就是一个数据页的大小,是固定的,如果数据项占的空间越小,数据项的数量越多,树的高度越低。这就是为什么每个数据项,即索引字段要尽量的小,比如int占4字节,要比bigint8字节少一半。这也是为什么b+树要求把真实的数据放到叶子节点而不是内层节点,一旦放到内层节点,磁盘块的数据项会大幅度下降,导致树增高。当数据项等于1时将会退化成线性表。

2.索引的最左匹配特性:当b+树的数据项是复合的数据结构,比如(name,age,sex)的时候,b+数是按照从左到右的顺序来建立搜索树的,比如当(张三,20,F)这样的数据来检索的时候,b+树会优先比较name来确定下一步的所搜方向,如果name相同再依次比较age和sex,最后得到检索的数据;但当(20,F)这样的没有name的数据来的时候,b+树就不知道下一步该查哪个节点,因为建立搜索树的时候name就是第一个比较因子,必须要先根据name来搜索才能知道下一步去哪里查询。比如当(张三,F)这样的数据来检索时,b+树可以用name来指定搜索方向,但下一个字段age的缺失,所以只能把名字等于张三的数据都找到,然后再匹配性别是F的数据了, 这个是非常重要的性质,即索引的最左匹配特性。

聚集索引

默认每张表主键会生成一个聚集索引,聚集索引会在叶子几点直接存放数据,不用回表查询,速度最快

#InnoDB存储引擎表是索引组织表,即表中数据按照主键顺序存放。
而聚集索引(clustered index)就是按照每张表的主键构造一棵B+树,同时叶子结点存放的即为整张表的行记录数据,也将聚集索引的叶子结点称为数据页。
聚集索引的这个特性决定了索引组织表中数据也是索引的一部分。同B+树数据结构一样,每个数据页都通过一个双向链表来进行链接。
    
#如果未定义主键,MySQL取第一个唯一索引(unique)而且只含非空列(NOT NULL)作为主键,InnoDB使用它作为聚簇索引。
    
#如果没有这样的列,InnoDB就自己产生一个这样的ID值,它有六个字节,而且是隐藏的,使其作为聚簇索引。

#由于实际的数据页只能按照一棵B+树进行排序,因此每张表只能拥有一个聚集索引。
在多数情况下,查询优化器倾向于采用聚集索引。因为聚集索引能够在B+树索引的叶子节点上直接找到数据。
此外由于定义了数据的逻辑顺序,聚集索引能够特别快地访问针对范围值得查询。

非聚集索引

新建的其他普通索引都是非聚集索引,而是存放这一行主键的值,需要回表查询

辅助索引的存在并不影响数据在聚集索引中的组织,因此每张表上可以有多个辅助索引,但只能有一个聚集索引。当通过辅助索引来寻找数据时,InnoDB存储引擎会遍历辅助索引并通过叶子级别的指针获得只想主键索引的主键,然后再通过主键索引来找到一个完整的行记录。

举例来说,如果在一棵高度为3的辅助索引树种查找数据,那需要对这个辅助索引树遍历3次找到指定主键,如果聚集索引树的高度同样为3,那么还需要对聚集索引树进行3次查找,最终找到一个完整的行数据所在的页,因此一共需要6次逻辑IO访问才能得到最终的一个数据页。

索引合并

where的两个条件列都建立了索引,会临时合并成一条索引Using union(ind_id,index_email)

create index ind_id on s1(id)
create index ind_email on s1(email)
select * from s1 where id=100 or email = 'eva100@oldboy' # eva100@oldboy的id就是一百


mysql> explain select * from s1 where id=1000 or email = 'eva100@oldboy';
+----+-------------+-------+-------------+--------------------+--------------------+---------+------+------+----------------------------------------------+
| id | select_type | table | type        | possible_keys      | key                | key_len | ref  | rows | Extra                                        |
+----+-------------+-------+-------------+--------------------+--------------------+---------+------+------+----------------------------------------------+
|  1 | SIMPLE      | s1    | index_merge | ind_id,index_email | ind_id,index_email | 5,153   | NULL |    2 | Using union(ind_id,index_email); Using where |
+----+-------------+-------+-------------+--------------------+--------------------+---------+------+------+----------------------------------------------+

覆盖索引:在查询过程中不需要回表 Using index

select id from s1 where id =100     覆盖索引:在查找一条数据的时候,命中索引,不需要再回表
 select name from s1 where id =100   相对慢

索引的创建和删除

create index ind_name on 表名(字段名);
create index ind_name on 表名(字段名,字段2);
drop index 索引名 on 表名

正确的使用索引

# 正确的使用mysql数据库
    # 从库的角度
        # 搭建集群
        # 读写分离
        # 分库
    # 从表的角度
        # 合理安排表与表之间的关系 :该拆的拆,该合的合
        # 把固定长度的字段放在前面
        # 尽量使用char而不是varchar
    # 从操作数据的角度
        # 尽量在where字段就约束数值到一个比较小的范围 : 分页
            # where a between value1 and value2
        # 尽量使用连表查询代替子查询
        # 删除数据和修改数据的时候条件尽量使用主键
        
        # 合理的创建和使用索引
            # 创建索引
                # 1.选择区分度比较大的列
                # 2.尽量选择短的字段创建索引
                # 3.不要创建不必要的索引,及时删除不用的索引,索引太多会拖慢写入数据的速度
                
            # 使用索引
                # 1.查询的字段不是索引字段
                # 2.在条件中使用范围,结果的范围越大速度越慢,范围小就快
                # 3.like 'a%'命中索引,like '%a'不命中索引
                # 4.条件列不能参与计算\不能使用函数
                # 5.and/or
                    # and条件相连 有一列有索引都会命中
                    # or条件相连 所有列都有索引才能命中
                # 6.联合索引
                    # create index mix_ind on 表 (id,name,email)
                    # 遵循最左前缀原则,且从出现范围开始索引失效
                    # select * from 表 where id = 123; 命中索引
                    # select * from 表 where id > 123; 不命中索引
                    # select * from 表 where id = 123 and name = 'alex'; 命中索引
                    # select * from 表 where id > 123 and name = 'alex'; 不命中索引
                    # select * from 表 where id = 123 and email = 'alex@oldboy'; 命中索引
                    # select * from 表 where email = 'alex@oldboy'; 不命中索引,因为条件中没有id
                    # select * from 表 where name='alex' and email = 'alex@oldboy'; 不命中索引,因为条件中没有id
                # 7.条件中的数据类型和实际字段的类型必须一致
                # 8.select字段中应该包含order by 中的字段
                    # select age from 表 order by age;   快
                    # select name from 表 order by age;  慢
                    

# 其他注意事项
- 避免使用select *
- 使用count(*)
- 创建表时尽量使用 char 代替 varchar
- 表的字段顺序固定长度的字段优先
- 组合索引代替多个单列索引(由于mysql中每次只能使用一个索引,所以经常使用多个条件查询时更适合使用组合索引)
- 尽量使用短索引
- 使用连接(JOIN)来代替子查询(Sub-Queries)
- 连表时注意条件类型需一致
- 索引散列值(重复少)不适合建索引,例:性别不适合

https://www.cnblogs.com/Eva-J/articles/10126413.html#_label7

https://www.cnblogs.com/Eva-J/articles/9772614.html

标签:index,name,18,索引,mysql,where,id,select
From: https://www.cnblogs.com/ejjw/p/17930624.html

相关文章

  • 即时通讯技术文集(第29期):IM开发技术合集(Part2) [共18篇]
    为了更好地分类阅读52im.net总计1000多篇精编文章,我将在每周三推送新的一期技术文集,本次是第29 期。[- 1 -] 谈谈移动端IM开发中登录请求的优化[链接] http://www.52im.net/thread-282-1-1.html[摘要] 到底是“登陆”还是“登录”?这是很多处女坐开发者纠结的问题,不......
  • 17--MySQL优化
    MySQL优化一、优化工具:1、系统优化工具1.1 top (1)简介: 实时监控当前操作系统的负载情况的,每秒刷新一次状态,通常会关注三大指标(CPU、MEM、IO) (2)评判标准 (2.1) 整体的负载情况,判断标准,如果值非常高,只能告诉我们操作系统很繁忙 load average: 0.00, 0.00, 0.00 ......
  • 11-MySQL 存储引擎
    MySQL存储引擎可以理解为,MySQL的“文件系统”,(插件形式存在)只不过功能更加强大。mysql提供的存储引擎有InnoDB、MyISAM等TokuDB是第三方的存储引擎moardb,数据压缩比打,写入数据快,如果你的应用是读多写少的情况强烈建议使用此存储引擎innodb存储引擎的功能一、查看存储引擎1......
  • 13-Mysql备份恢复
    一、备份的类型冷备份:关闭数据、停止业务温备份:加锁备份热备份:在先备份,不会影响到也正常运行二、备份方式2.1逻辑备份:基于SQL语句的备份mysqldump建库、建表、数据插入基于二进制日志:数据库的所有变化类的操作基于复制的备份:将二进制日志实时传送到另一台机器并且恢复......
  • 面试官:MySQL 到底是 join 性能好,还是 in 一下更快呢?被问懵逼了…
    来源:https://juejin.cn/post/7169567387527282701先总结:数据量小的时候,用join更划算数据量大的时候,join的成本更高,但相对来说join的速度会更快数据量过大的时候,in的数据量过多,会有无法执行SQL的问题,待解决事情是这样的,去年入职的新公司,之后在代码review的时候被提出说,不要......
  • 12-Mysql的日志管理
    一、mysql常见日志二、错误日志配置方法:[mysqld]log-error=/data/mysql/mysql.log查看配置方式:mysql>showvariableslike'%log%error%';作用:记录mysql数据库的一般状态信息及报错信息,是我们对于数据库常规报错处理的常用日志,默认在data目录下三、一般查询......
  • 14-Mysql主从复制
    一、mysql复制原理1.1主从复制原理过程从库的I/Othread线程会读取masterinfo文件获取主库的user,passwordport信息然后还会获取上次获取主库二进制日志的位置如3640就是00003这个文件640这个位置,主库收到从库的请求后,会验证用户名密码等的合法性,然后问主库你有没有......
  • 04-Mysql多实例
    多实例就是多套线程和多各进程和多个预分配的内存结构配置思路启动多个mysqld进程规划多套数据规划多个端口规划多套日志路径配置例子1、创建多套目录mkdir-p/data/330{7,8,9}2、准备多套配置文件vi/data/3307/my.cnf[mysqld]basedir=/application/mysqldatadi......
  • 05-Mysql 用户管理
    一、MySQL用户管理用户定义:user主机范围使用某个用户从哪个(些)地址访问我的数据库用户的功能:1、用来登录mysql数据库2、用来管理数据库对象(库、表)权限功能:针对不同用户设置对不同对象管理能力selectupdatedeleteinsertcreatedrop。。。权限范围:......
  • 01-Mysql介绍及安装
    关系型数据库的特点二维表典型产品Oracle传统企业,MySQL是互联网企业数据存取是通过SQL最大特点,数据安全性方面强(ACID)•NoSQL:非关系型数据库(NotonlySQL)不是否定关系型数据库,做关系型数据库的的补充想做老大,先学会做老二•NoSQL特性总览–不是否定......