mysql优化的几种方法
思考sql优化的几个地方,我把他做了个分类,方便理解
select [字段 优化1]:主要是覆盖索引
from []
where [条件 优化2]
union [联合查询 优化3]
新建表格
CREATE TABLE `student` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
`name` varchar(50) DEFAULT NULL COMMENT '姓名',
`age` int(11) DEFAULT NULL COMMENT '年龄',
`phone` varchar(12) DEFAULT NULL,
`create_time` datetime DEFAULT NULL COMMENT '创建时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
添加索引,添加索引之后
key_len:根据这个值,就可以判断索引使用情况,特别是在组合索引的时候,判断所有的索引字段是否都被查询用到。
key_len计算方式简单介绍
latin1占用1个字节,gbk占用2个字节,utf8占用3个字节
不允许为空:
- varchar(10):10*3
- char(10):10*3+2
- int:4
允许为空:
- varchar(10):10*3+1
- char(10):10*3+2+1
- int:4+1
使用完全索引key_len=name(50*3+2+1=153)+age(4+1)+phone(12*3+2+1=39)
alter table studen add index name_age_phone(name, age, phone);
添加数据
insert into student(name,age,phone,create_time) values('赛文',1000,'15717177664',now());
insert into student(name,age,phone,create_time) values('雷欧',1200,'15733337664',now());
insert into student(name,age,phone,create_time) values('泰罗',800,'15714447664',now());
一、优化点1:字段优化
覆盖索引尽量用
简单解释解释,索引是哪几个列,就查询哪几个列: 覆盖索引的原因:索引是高效找到行的一个方法,但是一般数据库也能使用索引找到一个列的数据,因此它不必读取整个行。毕竟索引页子节点存储了它们索引的数据; 当能通过读取索引就可以得到想要的数据,那就不需要读取行了。一个索引包含了(或覆盖了)满足查询结果的数据就叫做覆盖索引 注意:有索引尽量不要使用select *
#未覆盖索引
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗' and age =1000 and phone='15717177664';
#覆盖了索引
EXPLAIN SELECT name,age,phone FROM student WHERE NAME = '泰罗' and age =1000 and phone='15717177664';
#包含了索引
EXPLAIN SELECT name FROM student WHERE NAME = '泰罗' and age =1000 and phone='15717177664';
#加上主键也还是覆盖索引
EXPLAIN SELECT id, name,age,phone FROM student WHERE NAME = '泰罗' and age =1000 and phone='15717177664';
未使用覆盖索引
data:image/s3,"s3://crabby-images/521f6/521f6ffb24fd56183f21c0a8176875be06a7974c" alt=""
使用完全覆盖索引
data:image/s3,"s3://crabby-images/ed763/ed7632d7175a33ba23d2cd832233684d721f27b3" alt=""
使用包含覆盖索引
data:image/s3,"s3://crabby-images/b0075/b0075d2c39240ccf8c12d8beaa4dbcbd20c08416" alt=""
加上主键还是覆盖索引
data:image/s3,"s3://crabby-images/7a0a0/7a0a0cbcf36ff651c6f0245cc64d66404c1133db" alt=""
二、优化点2:where优化
1.尽量全值匹配
EXPLAIN SELECT * FROM student WHERE NAME = '赛文';
EXPLAIN SELECT * FROM student WHERE NAME = '雷欧' AND age = 1200;
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗' AND age = 800 AND phone = '15714447664';
执行结果,三个都用到了索引,但是key_len是不同的,key_len=197,表示所有索引都使用到了
data:image/s3,"s3://crabby-images/5e208/5e208ba3490da660688248b7e0033a5102af2f66" alt=""
当建立了索引列后,能在 wherel条件中使用索引的尽量所用。
2.最佳左前缀法则
最左前缀法则:指的是查询从索引的最左前列开始并且不跳过索引中的列。
我们定义的索引顺序是name_age_phone,所以查询的时候也应该从name开始,然后age,然后phone
情况1:从age、phone开始查询,tpye=All,key = null,没使用索引
data:image/s3,"s3://crabby-images/a3787/a3787b665a513500e257a10ada7f7b5fd0e20acc" alt=""
情况2:从phone开始查询,type=All,key=null,未使用索引
data:image/s3,"s3://crabby-images/8d5eb/8d5eb3d99bdf62378cad71270abb1a924bc1ed86" alt=""
data:image/s3,"s3://crabby-images/55634/55634c87d175a72b6952934b86d8b8c91dc76743" alt=""
情况3:从name开始,type=ref,使用了索引
data:image/s3,"s3://crabby-images/55634/55634c87d175a72b6952934b86d8b8c91dc76743" alt=""
3.范围条件放最后
没有使用范围查询,key_len=197,使用到了name+age+phone组合索引
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗' AND age = 1000 AND phone = '15717177664';
data:image/s3,"s3://crabby-images/6d811/6d811e9c59fcdc803b686a72f5435d7d91ee45b4" alt=""
使用了范围查询,key_len从197变为158,即除了name和age,phone索引失效了
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗' AND age > 800 AND phone = '15717177664';
key_len=name(153)+age(5)
data:image/s3,"s3://crabby-images/04ce3/04ce3b9dd4030c96d2ddb4b8ee9651fd7f0d298b" alt=""
4.不在索引列上做任何操作
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗';
EXPLAIN SELECT * FROM student WHERE left(NAME,1) = '泰罗';
不做计算,key_len有值,key_len=153,有使用name索引
data:image/s3,"s3://crabby-images/77fff/77fffaaf7e7ed7da93323c0850c25dec11e8ecc4" alt=""
做了截取结算,type=All,key_len=null,未使用索引
data:image/s3,"s3://crabby-images/7078c/7078c95189342c29796098bef59f97ab451c181f" alt=""
5.不等于要甚用
mysql 在使用不等于(!= 或者<>)的时候无法使用索引会导致全表扫描
#有使用到索引
EXPLAIN SELECT * FROM student WHERE NAME = '泰罗';
#不等于查询,未使用到索引
EXPLAIN SELECT * FROM student WHERE NAME != '泰罗';
EXPLAIN SELECT * FROM student WHERE NAME <> '泰罗';
#如果定要需要使用不等于,请用覆盖索引
EXPLAIN SELECT name,age,phone FROM student WHERE NAME != '泰罗';
EXPLAIN SELECT name,age,phone FROM student WHERE NAME <> '泰罗';
使用不等于查询,跳过索引
data:image/s3,"s3://crabby-images/13cfe/13cfea4812440f3bd8a3cfe35ef86b37928e2039" alt=""
使用不等于查询,同时使用覆盖索引,此时可以使用到索引
data:image/s3,"s3://crabby-images/cbd84/cbd8450bc1ff16bd2eedbbb04a1dee1a235f70cf" alt=""
6.Null/Not null有影响
修改为非空
data:image/s3,"s3://crabby-images/d7cef/d7cefda1f7ada3676b78e71b79492b484c2c805c" alt=""
data:image/s3,"s3://crabby-images/c8e36/c8e36fe877a9c82f62c7148c516663fcc660b494" alt=""
那么为not null,此时导致索引失效
EXPLAIN select * from student where name is null;
EXPLAIN select * from student where name is not null;
data:image/s3,"s3://crabby-images/12784/12784e2b05e8fbd1a09c64eb1fda805d60f7f8fd" alt=""
data:image/s3,"s3://crabby-images/bfd51/bfd512480dd421e4e25f105e0ae01b21984344ca" alt=""
改为可以为空
data:image/s3,"s3://crabby-images/5b118/5b118b4a3cfdc8473578e6b968ee0382b1519a15" alt=""
查询为空,索引起作用了
data:image/s3,"s3://crabby-images/f596f/f596fece80d87db854c5a807ae7a2ad7276edb32" alt=""
查询非空索引失效
data:image/s3,"s3://crabby-images/d307c/d307c9bd11569570db7dfe94fd5b30312489becf" alt=""
解决方法:
使用覆盖索引(覆盖索引解千愁)
data:image/s3,"s3://crabby-images/a6711/a6711b33bc6a299d651c8fb4662b4ed2b5d7c5f6" alt=""
7、Like 查询要当心
like 以通配符开头('%abc...')mysql 索引失效会变成全表扫描的操作
#like 以通配符开头('%abc...')mysql 索引失效会变成全表扫描的操作
#索引有效
EXPLAIN select * from student where name ='泰罗';
#索引失效
EXPLAIN select * from student where name like '%泰罗%';
#索引失效
EXPLAIN select * from student where name like '%泰罗';
#索引有效
EXPLAIN select * from student where name like '泰罗%';
解决方式:覆盖索引
EXPLAIN select name,age,phone from student where name like '%泰罗%';
data:image/s3,"s3://crabby-images/b3546/b354635fc171aa4792930131e66c3df67c802dde" alt=""
data:image/s3,"s3://crabby-images/2e608/2e6087db3deff5a0d40f20cb2a6629dd9c4c9d66" alt=""
data:image/s3,"s3://crabby-images/b65e4/b65e496942a0c877ab6671942d37a4b07b281892" alt=""
使用覆盖索引能够解决
data:image/s3,"s3://crabby-images/f1b34/f1b347ff5e4ae1de2f95f0efce6ede80e992b1cf" alt=""
8.字符类型加引号
字符串不加单引号索引失效(这个看着有点鸡肋了,一般查询字符串都会加上引号)
#不加引号导致索引失效
EXPLAIN select * from student where name =11;
EXPLAIN select * from student where name ='泰罗';
data:image/s3,"s3://crabby-images/ec410/ec410536b3dd59e70f177d444a09230a74331f9c" alt=""
使用覆盖索引解决
data:image/s3,"s3://crabby-images/ed1d4/ed1d4924825b9469f48b85681d76db478217eaf4" alt=""
三、优化3
1.OR 改 UNION 效率高
未使用索引
EXPLAIN select * from student where name='泰罗' or name = '雷欧';
使用索引
EXPLAIN
select * from student where name='泰罗'
UNION
select * from student where name = '雷欧';
解决方式:覆盖索引
EXPLAIN select name,age from student where name='泰罗' or name = '雷欧';
data:image/s3,"s3://crabby-images/6e3ba/6e3bac9cb0f3ea330d591ef2fd6f1b7f237a0ea3" alt=""
使用or未使用到索引
data:image/s3,"s3://crabby-images/3cbfd/3cbfd1393ce5e43fedd6b9430128efed32a70347" alt=""
data:image/s3,"s3://crabby-images/b1fe5/b1fe50acd15912eb34a0f9ce60312d66f4b44dd3" alt=""
使用union,使用了索引
解决方式:覆盖索引
data:image/s3,"s3://crabby-images/8664d/8664daee64a4ae815e06ddef25befccf2585f9f1" alt=""