python之路_mysql数据库索引相关

  索引本质都是:经过不断地缩小想要获取数据的范围来筛选出最终想要的结果,同时把随机的事件变成顺序的事件,也就是说,有了这种索引机制,咱们能够老是用同一种查找方式来锁定数据。索引的主要的功能就是加速查找。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);

日志管理
相关文章
相关标签/搜索