当写完一个sql,如果不能确定该语句的执行情况,建议使用EXPLAIN关键字,根据执行结果,进行优化分析。EXPLAIN关键字详情
- in包含的值不要过多
Oracle中,in语句中可放的最大参数个数是1000个,mysql中,in语句中参数个数是不限制的。不过对整段sql语句的长度有了限制(max_allowed_packet默认位4M),但是最好不要超过200个值(对于in中包含值数量的性能测试)。 - select语句指明字段名称
如果使用了select * 增加了不必要的io或者说是回表(为什么不建议用select *) - 排序字段要建立索引
数据库中的记录都是按照集合的方式来组织的,一个记录集中的各个记录往往是无序的(或者有默认顺序,如按存入数据库的顺序,但不能绝对保证这种顺序),而索引是有序的,所以使用索引能大大加快查询速度 - 避免在where子句中对字段进行null值判断
对于null的判断可能会导致引擎放弃使用索引而进行全表扫描,为什么说是可能呢?这取决于mysql决定执不执行某个索引的成本够不够小。经过测试发现,假设数据记录条数不变,如果大部分数据都是null值,那么is null不会使用索引,会进行全表扫描,而is not null会使用到索引,反之,如果大部分不为空,is null 会使用索引,is not null不会使用索引。(Mysql中null值到底是否用到索引) - 不使用%前缀模糊查询
如like %name或者like %name%,这种也会导致索引失效,而 like name%是会走索引的 - 避免隐式类型转换
where子句中出现column字段的类型和传入的参数类型不一致的时候发生的类型转换,建议先确定where中的参数类型。 - 联合索引,遵循最左前缀法则
举列来说联合索引含有字段id、name、school,可以直接用id字段,也可以id、name这样的顺序,都会走索引的,但是name;school都无法使用这个索引。所以在创建联合索引的时候一定要注意索引字段顺序,常用的查询字段放在最前面。(最左前缀法则)
假设 key1,key2,key3为联合索引 select key1,key2,key3 from person where key2 = 'boss',这种情况大家是不是会认为不走索引呢?其实它也是会走索引的,因为它不需要回表,直接可以在这个二级索引(联合索引)里面获得值。 - 不要在列上进行运算
如果查询条件中含有函数或者表达式,也会导致索引失效
例如 select * from user where YEAR(birthday) < 2000 - 可使用force index来强制走某个索引
有的时候MySQL优化器采取它认为合适的索引来检索SQL语句,但是可能它所采用的索引并不是我们想要的。这时就可以采用forceindex来强制优化器使用我们指定的索引。