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