上一篇博客讲了可使用慢查询日志定位耗时sql,使用explain命令查看mysql的执行计划,以及使用profiling工具查看语句执行真正耗时的地方,当定位了耗时以后怎样优化呢?这篇博客会介绍mysql中最简单快速的优化方法——添加索引。mysql
1、索引的添加 sql
mysql一共有四类索引,分别是主键索引、惟一索引、普通索引以及全文索引。数据库
1.一、主键索引的添加 工具
建立一张表时,把某个列设为主键的时候,则该列就是主键索引。添加主键索引有两种方法,咱们能够在建表的时候指定主键,这样就能够在建表时添加索引;咱们也能够在建表之后添加主键索引,下面的table1的主键索引是在建表时建立,table2是在建表后添加的主键索引:优化
create table table1( id int unsigned primary key auto_increment , name varchar(32) not null ); //添加主键索引语法:alter table 表名 add primary key (列名); create table table2( id int unsigned, name varchar(32) not null ); alter table table2 add primary key (id);
1.二、惟一索引的添加spa
当表的某列被指定为unique约束时,这列就是一个惟一索引。一样,惟一索引的添加也能够分建立时添加和建立后添加,下例中table3的惟一索引是建立时添加,table4的惟一索引是建立后添加的:日志
create table table3( id int unsigned primary key auto_increment , name varchar(32) unique ); //添加惟一索引语法:create unique index 索引名 on 表名 (列表..); create table table4( id int unsigned primary key auto_increment , name varchar(32) ); create unique index my_uni on table4(name);
1.三、普通索引的添加blog
普通索引应该是mysql中最经常使用的索引,由于一张表只能有一个主键索引,数量较少,惟一索引又必须保证内容惟一,限制较多,因此在mysql中最经常使用的仍是普通索引。普通索引的建立也能够分建表时建立和建表后建立,下面例子中table5的普通索引是建表时建立,table6是的普通索引是建表后建立的: 排序
create table table5( id int unsigned, name varchar(32), KEY KEY_ID_NAME(id, name) ); //添加普通索引语法:create index 索引名 on 表 (列1,列名2); create table table6( id int unsigned, name varchar(32) ); create index KEY_ID_NAME on table6 (id, name);
1.四、全文索引索引
全文索引主要是针对文本或者文件的搜索,在这里就再也不举例介绍了。
2、索引的查询
索引的查询很简单,可使用如下两条命令显示表上的索引:
show index from 表名 或 show keys from 表名
以table5为例,咱们能够看看怎么查询索引:
mysql> show index from table5\G; *************************** 1. row *************************** Table: table5 Non_unique: 1 Key_name: KEY_ID_NAME Seq_in_index: 1 Column_name: id Collation: A Cardinality: NULL Sub_part: NULL Packed: NULL Null: YES Index_type: BTREE Comment: *************************** 2. row *************************** Table: table5 Non_unique: 1 Key_name: KEY_ID_NAME Seq_in_index: 2 Column_name: name Collation: A Cardinality: NULL Sub_part: NULL Packed: NULL Null: YES Index_type: BTREE Comment: 2 rows in set (0.00 sec)
从查询结果能够看出,索引类型是B树,两列结果的Key_name同样,说明是复合索引等等信息。不过这样看很累,在平时查看索引时,我喜欢使用'show create table'命令查看表上的索引:
mysql> show create table table5\G; *************************** 1. row *************************** Table: table5 Create Table: CREATE TABLE `table5` ( `id` int(10) unsigned DEFAULT NULL, `name` varchar(32) DEFAULT NULL, KEY `KEY_ID_NAME` (`id`,`name`) ) ENGINE=MyISAM DEFAULT CHARSET=latin1 1 row in set (0.00 sec)
这样看结果就清晰多了,'KEY KEY_ID_NAME(id, name)'直接把索引名称,索引做用列以及索引类型都显示出来了,清晰明了。
3、索引的删除
可使用如下命令删除索引:alter table 表名 drop index 索引名; 例如咱们想删除table5上的KEY_ID_NAME索引只需使用如下命令便可:
alter table table5 drop index KEY_ID_NAME;
比较特殊的是删除主键索引,删除主键索引的语法以下:alter table tablename drop primary key; 例如咱们要删除table2上的主键索引可使用如下命令:
alter table table2 drop primary key;
4、索引的利弊
索引为咱们带来的好处主要有如下两点:一、缩小检索范围,加快检索速度;二、在索引上排序及group by资源消耗极低。
同时,大量使用索引也会给咱们带来如下弊端:一、增删改操做将比原来更加耗时;二、索引的存储会占用存储空间。
5、建立索引的考虑点
当要为表建立索引时,咱们能够从如下几个方面考虑是否应该建立索引:
一、较频繁的做为查询条件的字段应该建立索引;
二、惟一性太差的字段不适合单首创建索引,即便频繁做为查询条件;
三、更新很是频繁的字段不适合建立索引,由于添加索引后字段更新将更加耗时;
6、使用索引的注意项
一、对于建立的多列索引,只要查询条件使用了最左边的列,索引通常就会被使用。例如在a,b,c列上建立索引,那么查询条件"a=5","a=5 and b=6","a=7 and b=8 and c=6"都会使用该索引;
二、对于使用like的查询,查询若是是‘%aaa’不会使用到索引‘aaa%’会使用到索引;
三、若是条件中有or,那么除非or条件都带有索引,不然仍是会全表扫描;
四、若是列类型是字符串,那必定要在条件中将数据使用引号引用起来,不然不使用索引;
五、若是字符串较长时,直接使用'create index KEY_NAME on table6 (name)' 命令建立的索引也将较长,浪费磁盘空间;这时候咱们能够经过'create index KEY_NAME on table6 (name(4))' 命令限定索引长度;
六、链接查询时,在链接键上增长索引能够加快速度,同时遵循小表驱动大表原则;
七、若是mysql估计使用全表扫描要比使用索引快,则不使用索引。
7、直观感觉索引的威力
上一篇博客中咱们建立了一张包含1000w数据的数据表,在该表上咱们没有加任何索引,而后查询一条数据话费了6.5s时间。此次咱们依然构造一张包含1000w数据的一样结构数据表,不一样的是咱们为它加上索引,而后查询数据,看一下索引优化的效果。
首先,建立数据库表:
CREATE TABLE emp (
empno MEDIUMINT UNSIGNED NOT NULL DEFAULT 0 COMMENT '编号', ename VARCHAR(20) NOT NULL DEFAULT "" COMMENT '名字', job VARCHAR(9) NOT NULL DEFAULT "" COMMENT '工做', mgr MEDIUMINT UNSIGNED NOT NULL DEFAULT 0 COMMENT '上级编号', hiredate DATE NOT NULL COMMENT '入职时间', sal DECIMAL(7,2) NOT NULL COMMENT '薪水', comm DECIMAL(7,2) NOT NULL COMMENT '红利', deptno MEDIUMINT UNSIGNED NOT NULL DEFAULT 0 COMMENT '部门编号', KEY KEY_NO(empno) )ENGINE=InnoDB DEFAULT CHARSET=utf8;
而后调用存储过程插入1000w条数据:
call insert_emp(10000000);
最后,调用和上篇博客如出一辙的查询语句,看此次查询须要多长时间:
mysql> select * from emp where empno=413345; +--------+--------+----------+-----+------------+---------+--------+--------+ | empno | ename | job | mgr | hiredate | sal | comm | deptno | +--------+--------+----------+-----+------------+---------+--------+--------+ | 413345 | JpugvK | SALESMAN | 1 | 2014-10-27 | 2000.00 | 400.00 | 157 | +--------+--------+----------+-----+------------+---------+--------+--------+ 1 row in set (0.10 sec)
查询时间从原来的6.5s提高到了0.1s,提升了65倍。