索引本质都是:经过不断地缩小想要获取数据的范围来筛选出最终想要的结果,同时把随机的事件变成顺序的事件,也就是说,有了这种索引机制,咱们能够老是用同一种查找方式来锁定数据。索引的主要的功能就是加速查找。html
1、mysql的常见索引mysql
普通索引INDEX:加速查找 惟一索引: -主键索引PRIMARY KEY:加速查找+约束(不为空、不能重复) -惟一索引UNIQUE:加速查找+约束(不能重复) 联合索引: -PRIMARY KEY(id,name):联合主键索引 -UNIQUE(id,name):联合惟一索引 -INDEX(id,name):联合普通索引
除此以外还有全文索引,即FULLTEXT,但其实对于全文搜索,咱们并不会使用MySQL自带的该索引,而是会选择第三方软件如Sphinx,专门来作全文搜索。sql
2、索引类型数据库
索引主要包括hash和btree两大类型,咱们在建立索引时能够为其指定索引类型。其中hash类型的索引:查询单条快,范围查询慢;btree类型的索引:b+树,层数越多,数据量指数级增加(咱们就用它,由于innodb默认支持它)。vim
#不一样的存储引擎支持的索引类型也不同 InnoDB 支持事务,支持行级别锁定,支持 B-tree、Full-text 等索引,不支持 Hash 索引; MyISAM 不支持事务,支持表级别锁定,支持 B-tree、Full-text 等索引,不支持 Hash 索引; Memory 不支持事务,支持表级别锁定,支持 B-tree、Hash 等索引,不支持 Full-text 索引; NDB 支持事务,支持行级别锁定,支持 Hash 索引,不支持 B-tree、Full-text 等索引; Archive 不支持事务,支持表级别锁定,不支持 B-tree、Hash、Full-text 等索引;
3、建立与删除索引服务器
一、在建立表时建立索引数据结构
create table t1( id int, name char(5), age int; unique key uni_name(name), #uni_name为索引名 index index_age(age), #index_age为索引名,不须要key primary key(id) #primary不须要起索引名,起了也不会显示 );
二、建立完表后为其添加索引函数
create table t3( id int, name char(5), age int ); create index indx_name on t3(name); #经常使用 alter table t3 add index indx_id(id); alter table t3 add primary key(age);
三、删除索引测试
drop index indx_id on t3;
alter table t3 drop primary key;
上述第一个删除语法中,因primary key 没有名字,因此删除方式为:drop index ‘primary’ on t3,其余有名字的索引删除方式为:drop index 索引名 on 表名优化
4、测试索引
按照以下sql语句建立表s1,后续全部测试均基于此表:
#1. 准备表 create table s1( id int, name varchar(20), gender char(6), email varchar(50) ); #2. 建立存储过程,实现批量插入记录 delimiter $$ #声明存储过程的结束符号为$$ create procedure auto_insert1() BEGIN declare i int default 1; while(i<3000000)do insert into s1 values(i,'egon','male',concat('egon',i,'@oldboy')); set i=i+1; end while; END$$ #$$结束 delimiter ; #从新声明分号为结束符号 #3. 查看存储过程 show create procedure auto_insert1\G #4. 调用存储过程 call auto_insert1();
一、加索引能够加快查询效率,可是会下降写的效率
5、正确使用索引
并非说咱们建立了索引就必定会加快查询速度,若想利用索引达到预想的提升查询速度的效果,咱们在添加索引时,必须遵循如下问题。
一、范围问题,或者说条件不明确,条件中出现这些符号或关键字:>、>=、<、<=、!= 、between...and...、like
大于 小于
不等于
between...and
like
二、尽可能选择区分度高的字段做为索引,区分度是指的字段中数据的重复性,越重复,区分度变低
咱们编写存储过程为表s1批量添加记录,name字段的值均为egon,也就是说name这个字段的区分度很低(gender字段也是同样的,咱们稍后再搭理它) 回忆b+树的结构,查询的速度与树的高度成反比,要想将树的高低控制的很低,须要保证:在某一层内数据项均是按照从左到右,从小到大的顺序依次排开,即左1<左2<左3<... 而对于区分度低的字段,没法找到大小关系,由于值都是相等的,毫无疑问,还想要用b+树存放这些等值的数据,只能增长树的高度,字段的区分度越低,则树的高度越高。极端的状况,索引字段的值都同样,那么b+树几乎成了一根棍。本例中就是这种极端的状况,name字段全部的值均为'egon' #如今咱们得出一个结论:为区分度低的字段创建索引,索引树的高度会很高,然而这具体会带来什么影响呢??? #1:若是条件是name='xxxx',那么确定是能够第一时间判断出'xxxx'是不在索引树中的(由于树中全部的值均为'egon’),因此查询速度很快 #2:若是条件正好是name='egon',查询时,咱们永远没法从树的某个位置获得一个明确的范围,只能往下找,往下找,往下找。。。这与全表扫描的IO次数没有多大区别,因此速度很慢
三、索引字段不能够参与计算
四、and or
#注意: 条件1 and 条件2:查询原理是:首先条件1与条件2都成立的前提下,才算匹配成功一条记录;其次mysql会按先优先判断索引字段的条件,若是按照该条件为真,但锁定的范围很小,或者干脆为假,那咱们即使是没有为其余条件的字段添加索引,最终的结果仍然很快 #例如: 若条件1的字段有索引,而条件2的字段没有索引,那么若是在按照条件1查出的结果不多的状况下,即使咱们没有为条件2建立索引,最终的查询速度依然很快 若条件1的字段没有索引,而条件2的字段有索引,那么若是在按照条件2查出的结果不多的状况下,即使咱们没有为条件1建立索引,最终的查询速度依然很快
在左边条件成立可是索引字段的区分度低的状况下(name与gender均属于这种状况),会依次往右找到一个区分度高的索引字段,加速查询
通过分析,在条件为name='egon' and gender='male' and id>333 and email='xxx'的状况下,咱们彻底不必为前三个条件的字段加索引,由于只能用上email字段的索引,前三个字段的索引反而会下降咱们的查询效率。
五、最左前缀匹配原则,很是重要的原则,对于组合索引mysql会一直向右匹配直到遇到范围查询(>、<、between、like)就中止匹配(指的是范围大了,有索引速度也慢),好比a = 1 and b = 2 and c > 3 and d = 4 若是创建(a,b,c,d)顺序的索引,d是用不到索引的,若是创建(a,b,d,c)的索引则均可以用到,a,b,d的顺序能够任意调整。
六、其余状况
- 使用函数 select * from tb1 where reverse(email) = 'egon'; - 类型不一致 若是列是字符串类型,传入条件是必须用引号引发来,否则... select * from tb1 where email = 999; #排序条件为索引,则select字段必须也是索引字段,不然没法命中 - order by select name from s1 order by email desc; 当根据索引排序时候,select查询的字段若是不是索引,则速度仍然很慢 select email from s1 order by email desc; 特别的:若是对主键排序,则仍是速度很快: select * from tb1 order by nid desc; - 组合索引最左前缀 若是组合索引为:(name,email) name and email -- 命中索引 name -- 命中索引 email -- 未命中索引 - count(1)或count(列)代替count(*)在mysql中没有差异了 - create index xxxx on tb(title(19)) #text类型,必须制定长度
- 避免使用select * - count(1)或count(列) 代替 count(*) - 建立表时尽可能时 char 代替 varchar - 表的字段顺序固定长度的字段优先 - 组合索引代替多个单列索引(常用多个条件查询时) - 尽可能使用短索引 - 使用链接(JOIN)来代替子查询(Sub-Queries) - 连表时注意条件类型需一致 - 索引散列值(重复少)不适合建索引,例:性别不适合
6、索引合并与覆盖
一、索引合并
#索引合并:把多个单列索引合并使用 #分析: 组合索引能作到的事情,咱们均可以用索引合并去解决,好比 create index ne on s1(name,email);#组合索引 咱们彻底能够单独为name和email建立索引,而后按照where name='xxx' and email='xxx'使用 #索引合并 组合索引能够命中: select * from s1 where name='egon' ; select * from s1 where name='egon' and email='adf'; 索引合并能够命中: select * from s1 where name='egon' ; select * from s1 where email='adf'; select * from s1 where name='egon' and email='adf'; 乍一看好像索引合并更好了:能够命中更多的状况,但其实要分状况去看,若是是name='egon' and email='adf',那么组合索引的效率要高于索引合并,若是是单条件查,那么仍是用索引合并比较合理
二、索引覆盖
#覆盖索引: - 全部字段(条件的,查询结果的等)都是索引字段 http://blog.itpub.net/22664653/viewspace-774667/ #分析 select age from s1 where id=123 and name = 'egon'; #id字段有索引,可是name字段没有索引 该sql命中了索引,但未覆盖所有。 利用id=123到索引的数据结构中定位到了id字段,可是仍要判断name字段,可是name字段没有索引,并且查询结果的字段age也没有索引 最牛逼的状况是,索引字段覆盖了全部,那全程经过索引来加速查询以及获取结果就ok了
7、查询优化神器explain
关于explain命令相信你们并不陌生,具体用法和字段含义能够参考官网explain-output,这里须要强调rows是核心指标,绝大部分rows小的语句执行必定很快(有例外,下面会讲到)。因此优化语句基本上都是在优化rows。
8、慢日志管理
慢日志 - 执行时间 > 10 - 未命中索引 - 日志文件路径 配置: - 内存 show variables like '%query%'; show variables like '%queries%'; set global 变量名 = 值 - 配置文件 mysqld --defaults-file='E:\wupeiqi\mysql-5.7.16-winx64\mysql-5.7.16-winx64\my-default.ini' my.conf内容: slow_query_log = ON slow_query_log_file = D:/.... 注意:修改配置文件以后,须要重启服务
MySQL日志管理 ======================================================== 错误日志: 记录 MySQL 服务器启动、关闭及运行错误等信息 二进制日志: 又称binlog日志,以二进制文件的方式记录数据库中除 SELECT 之外的操做 查询日志: 记录查询的信息 慢查询日志: 记录执行时间超过指定时间的操做 中继日志: 备库将主库的二进制日志复制到本身的中继日志中,从而在本地进行重放 通用日志: 审计哪一个帐号、在哪一个时段、作了哪些事件 事务日志或称redo日志: 记录Innodb事务相关的如事务执行时间、检查点等 ======================================================== 1、bin-log 1. 启用 # vim /etc/my.cnf [mysqld] log-bin[=dir\[filename]] # service mysqld restart 2. 暂停 //仅当前会话 SET SQL_LOG_BIN=0; SET SQL_LOG_BIN=1; 3. 查看 查看所有: # mysqlbinlog mysql.000002 按时间: # mysqlbinlog mysql.000002 --start-datetime="2012-12-05 10:02:56" # mysqlbinlog mysql.000002 --stop-datetime="2012-12-05 11:02:54" # mysqlbinlog mysql.000002 --start-datetime="2012-12-05 10:02:56" --stop-datetime="2012-12-05 11:02:54" 按字节数: # mysqlbinlog mysql.000002 --start-position=260 # mysqlbinlog mysql.000002 --stop-position=260 # mysqlbinlog mysql.000002 --start-position=260 --stop-position=930 4. 截断bin-log(产生新的bin-log文件) a. 重启mysql服务器 b. # mysql -uroot -p123 -e 'flush logs' 5. 删除bin-log文件 # mysql -uroot -p123 -e 'reset master' 2、查询日志 启用通用查询日志 # vim /etc/my.cnf [mysqld] log[=dir\[filename]] # service mysqld restart 3、慢查询日志 启用慢查询日志 # vim /etc/my.cnf [mysqld] log-slow-queries[=dir\[filename]] long_query_time=n # service mysqld restart MySQL 5.6: slow-query-log=1 slow-query-log-file=slow.log long_query_time=3 查看慢查询日志 测试:BENCHMARK(count,expr) SELECT BENCHMARK(50000000,2*3); 日志管理