MySQL索引优化与分析(重要)

建表SQL

CREATE TABLE staffs (
  id INT PRIMARY KEY AUTO_INCREMENT,
  NAME VARCHAR (24)  NULL DEFAULT ‘‘ COMMENT ‘姓名‘,
  age INT NOT NULL DEFAULT 0 COMMENT ‘年龄‘,
  pos VARCHAR (20) NOT NULL DEFAULT ‘‘ COMMENT ‘职位‘,
  add_time TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT ‘入职时间‘
) CHARSET utf8 COMMENT ‘员工记录表‘ ;

INSERT INTO staffs(NAME,age,pos,add_time) VALUES(‘z3‘,22,‘manager‘,NOW());
INSERT INTO staffs(NAME,age,pos,add_time) VALUES(‘July‘,23,‘dev‘,NOW());
INSERT INTO staffs(NAME,age,pos,add_time) VALUES(‘2000‘,23,‘dev‘,NOW());
INSERT INTO staffs(NAME,age,pos,add_time) VALUES(null,23,‘dev‘,NOW());
SELECT * FROM staffs;

ALTER TABLE staffs ADD INDEX idx_staffs_nameAgePos(name, age, pos);

案例(索引失效)

1、全值匹配我最爱

索引  idx_staffs_nameAgePos 建立索引时 以 name , age ,pos 的顺序建立的。全值匹配表示 按顺序匹配的
EXPLAIN SELECT * FROM staffs WHERE NAME = ‘July‘;
EXPLAIN SELECT * FROM staffs WHERE NAME = ‘July‘ AND age = 25;
EXPLAIN SELECT * FROM staffs WHERE NAME = ‘July‘ AND age = 25 AND pos = ‘dev‘;

说明:如果查询过程中,若没有查询第一个NAME字段,索引将无效

有NAME,无age,部分索引失效

第一个索引字段相当于火车头,后面的索引字段相当于车厢,如果只剩下车头,火车依然可以开动,若没有车头,火车将无法运行。可以以此进行类比。

2、最佳左前缀法则

如果索引了多列,要遵守最左前缀法则。指的是查询从索引的最左前列开始并且不跳过索引中的列。

and 忽略左右关系。既即使没有没有按顺序 由于优化器的存在,会自动优化。
经过试验结论  建立了 idx_nameAge 索引  id 为主键
    1.当使用覆盖索引的方式时,(select name/age/id from staffs where age=10 (后面没有其他没有索引的字段条件)),即使不是以 name 开头,也会使用 idx_nameAge 索引。
    既 select 后的字段 有索引,where 后的字段也有索引,则无关执行顺序。
    2.除开上述条件 才满足最左前缀法则。

EXPLAIN SELECT * FROM staffs WHERE age = 25 AND pos = ‘dev‘;

EXPLAIN SELECT * FROM staffs WHERE pos = ‘dev‘;

口诀:带头大哥不能死,中间兄弟不能断。百分like加右边

3、不要在索引列上做任何操作(计算、函数、(自动or手动)类型转换),会导致索引失效而转向全表扫描

EXPLAIN SELECT * FROM staffs WHERE left(NAME,4) = ‘July‘;

4、存储引擎不能使用索引中范围条件右边的列

范围 若有索引则能使用到索引,范围条件右边的索引会失效(范围条件右边与范围条件使用的同一个组合索引,右边的才会失效。若是不同索引则不会失效)

5、尽量使用覆盖索引(只访问索引的查询(索引列和查询列一致)),减少select *

6、mysql 在使用不等于(!= 或者<>)的时候无法使用索引会导致全表扫描

索引  idx_nameAgeJob
         idx_name
使用 != 和 <> 的字段索引失效( != 针对数值类型。 <> 针对字符类型
前提 where and 后的字段在混合索引中的位置比比当前字段靠后  where age != 10 and name=‘xxx‘  ,这种情况下,mysql自动优化,将 name=‘xxx‘ 放在 age !=10 之前,name 依然能使用索引。只是 age 的索引失效)

7、is not null 也无法使用索引,但是is null是可以使用索引的

8、like以通配符开头(‘%abc...‘)mysql索引失效会变成全表扫描的操作

like ‘%abc%’ type 类型会变成 all
like ‘abc%’ type 类型为 range ,算是范围,可以使用索引

问题:解决like ‘%字符串%‘时索引不被使用的方法??

建表,插入数据

CREATE TABLE `tbl_user` (
 `id` INT(11) NOT NULL AUTO_INCREMENT,
 `NAME` VARCHAR(20) DEFAULT NULL,
 `age` INT(11) DEFAULT NULL,
 email VARCHAR(20) DEFAULT NULL,
 PRIMARY KEY (`id`)
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

#drop table tbl_user

INSERT INTO tbl_user(NAME,age,email) VALUES(‘1aa1‘,21,‘[email protected]‘);
INSERT INTO tbl_user(NAME,age,email) VALUES(‘2aa2‘,222,‘[email protected]‘);
INSERT INTO tbl_user(NAME,age,email) VALUES(‘3aa3‘,265,‘[email protected]‘);
INSERT INTO tbl_user(NAME,age,email) VALUES(‘4aa4‘,21,‘[email protected]‘);
INSERT INTO tbl_user(NAME,age,email) VALUES(‘aa‘,121,‘[email protected]‘);

建索引之前进行测试

#before index

EXPLAIN SELECT NAME,age    FROM tbl_user WHERE NAME LIKE ‘%aa%‘;

EXPLAIN SELECT id    FROM tbl_user WHERE NAME LIKE ‘%aa%‘;
EXPLAIN SELECT NAME     FROM tbl_user WHERE NAME LIKE ‘%aa%‘;
EXPLAIN SELECT age   FROM tbl_user WHERE NAME LIKE ‘%aa%‘;

EXPLAIN SELECT id,NAME    FROM tbl_user WHERE NAME LIKE ‘%aa%‘;
EXPLAIN SELECT id,NAME,age FROM tbl_user WHERE NAME LIKE ‘%aa%‘;
EXPLAIN SELECT NAME,age FROM tbl_user WHERE NAME LIKE ‘%aa%‘;

EXPLAIN SELECT *     FROM tbl_user WHERE NAME LIKE ‘%aa%‘;
EXPLAIN SELECT id,NAME,age,email  FROM tbl_user WHERE NAME LIKE ‘%aa%‘;

需求分析:查询时字段两边都是%,且索引不能失效

创建索引(覆盖索引)

#create index
CREATE INDEX idx_user_nameAge ON tbl_user(NAME,age);

#DROP INDEX idx_user_nameAge ON tbl_user

分析:email字段并没有创建索引,覆盖索引失效

覆盖索引:建立的索引字段和查询的字段在个数、顺序上最好完全一致。

9、字符串不加单引号索引失效

底层进行转换使索引失效,使用了函数造成索引失效

10、少用or,用它来连接时会索引失效

【总结】

带头大哥不能死

中间兄弟不能断(永远要符合最佳左前缀原则)

索引列上无计算

like百分加右边

范围之后全失效

字符串里有引号

原文地址:https://www.cnblogs.com/116970u/p/10987171.html

时间: 2024-08-27 13:26:21

MySQL索引优化与分析(重要)的相关文章

Mysql 索引优化分析

MySQL索引优化分析 为什么你写的sql查询慢?为什么你建的索引常失效?通过本章内容,你将学会MySQL性能下降的原因,索引的简介,索引创建的原则,explain命令的使用,以及explain输出字段的意义.助你了解索引,分析索引,使用索引,从而写出更高性能的sql语句.还在等啥子?撸起袖子就是干! 案例分析 我们先简单了解一下非关系型数据库和关系型数据库的区别. MongoDB是NoSQL中的一种.NoSQL的全称是Not only SQL,非关系型数据库.它的特点是性能高,扩张性强,模式灵

mySql索引优化分析

MySQL索引优化分析 为什么你写的sql查询慢?为什么你建的索引常失效?通过本章内容,你将学会MySQL性能下降的原因,索引的简介,索引创建的原则,explain命令的使用,以及explain输出字段的意义.助你了解索引,分析索引,使用索引,从而写出更高性能的sql语句.还在等啥子?撸起袖子就是干! 案例分析 我们先简单了解一下非关系型数据库和关系型数据库的区别.MongoDB是NoSQL中的一种.NoSQL的全称是Not only SQL,非关系型数据库.它的特点是性能高,扩张性强,模式灵活

重新学习MySQL数据库5:根据MySQL索引原理进行分析与优化

重新学习MySQL数据库5:根据MySQL索引原理进行分析与优化 一:Mysql原理与慢查询 MySQL凭借着出色的性能.低廉的成本.丰富的资源,已经成为绝大多数互联网公司的首选关系型数据库.虽然性能出色,但所谓"好马配好鞍",如何能够更好的使用它,已经成为开发工程师的必修课,我们经常会从职位描述上看到诸如"精通MySQL"."SQL语句优化"."了解数据库原理"等要求.我们知道一般的应用系统,读写比例在10:1左右,而且插入

MySQL索引优化步骤总结

在项目使用mysql过程中,随着系统的运行,发现一些慢查询,在这里总结一下mysql索引优化步骤 1.开发过程优化 开发过程中对业务表中查询sql分析sql执行计划(尤其是业务流水表),主要是查看sql执行计划,对sql进行优化. explain执行计划关键属性 select_type,possible_keys,key,rows (1) select_type 访问类型 system>const > eq_ref > ref > fulltext > ref_or_null

mysql索引优化

mysql 索引优化 >mysql一次查询只能使用一个索引.如果要对多个字段使用索引,建立复合索引. >越小的数据类型通常更好:越小的数据类型通常在磁盘.内存和CPU缓存中都需要更少的空间,处理起来更快. >简单的数据类型更好:整型数据比起字符,处理开销更小,因为字符串的比较更复杂.在MySQL中,应该用内置的日期和时间数据类型,而不是用字符串来存储时间:以及用整型数据类型存储IP地址. >尽量避免NULL:应该指定列为NOT NULL,除非你想存储NULL.在MySQL中,含有空

MySQL索引优化-from 高性能MYSQL

Btree: 1. 尽量使用覆盖索引, 即三星索引 2. 多列索引如果带范围的话, 后续列不会作为筛选条件 3. 多列索引应选择过滤性更好的充当前缀索引 4. 尽量按主键顺序插入, 减少页分裂, 采用自增ID在高并发情况下, 可能造成明显征用, 或者更改innodb_autoinc_lock_mode配置. Hash: 1.只有精确匹配所有列的查询才有效, 对于每行数据, 引擎都会对所有索引列计算hash码 2. 只有memory才可以支持hash索引, innodb支持自适应hash索引, 但

【转载】MySQL索引优化

MySQL索引优化 原文链接 MySQL官方对索引的定义:索引是帮助MySQL高效获取数据的数据结构.索引是在存储引擎中实现的,所以每种存储引擎中的索引都不一样.如MYISAM和InnoDB存储引擎只支持BTree索引:MEMORY和HEAP储存引擎可以支持HASH和BTREE索引. 这里仅针对常用的InnoDB存储引擎所支持的BTree索引进行介绍: 一.索引类型 先创建一个新表,用于演示索引类型 CREATE TABLE index_table ( id BIGINT NOT NULL au

Mysql索引优化分析-第一篇

1.性能下降SQL慢 执行时间长 等待时间长 查询语句写的烂 索引失效(单值,复合) 关联查询太多join(设计缺陷或不得已的需求) 服务器调优及各个参数设置(缓冲\线程数等) 2.常见通用的join查询 2.1SQL执行顺序 2.1.1手写 2.1.2机读 2.1.3总结 2.2Join图 2.3建表SQL 2.4 7种Join 3.索引简介 3.1什么是索引 MySQL官方对索引的定义为:索引(Index)是帮助MySQL高效获取数据的数据结构. 可以得到索引的本质:索引是数据结构 可以简单

Mysql 索引优化

索引的存储分类MyISAM 存储引擎的表数据和索引是自动分开存储的,各自是独一的一个文件Innodb 存储引擎的表数据和索引是存储在同一个表空间里面,但可以由多个文件构成.Mysql 目前不支持函数索引,但是能对列的前面某一部分进行索引例如 name 字段,可以只取 name 的前 4 个字符进行索引,可降低索引文件大小.Mysql> create index ind_company_name on company(name(4));说明:ind_company_name 索引名,company