MySQL高级知识-查询与索引优化分析

性能下降SQL慢、执行时间长、等待时间长

  • 查询语句写的烂
  • 索引失效
    • 单值索引
    • 复合索引
  • 关联查询太多join(设计缺陷或不得已的需求,除非你能干的过你的产品经理)
  • 服务器调优及各个参数设置(缓冲、线程数等)

常见通用的Join查询

SQL执行顺序
  • 手写
    • 《MySQL高级知识-查询与索引优化分析》 手写SQL顺序
SELECT DISTINCT
    <select_list>
FROM
     <left_table> <join_type>
JOIN <right_table> ON <join_condition>
WHERE 
    <where_condition>
GROUP BY
    <group_by_list>
HAVING
    <having_condition>
ORDER BY
    <order_by_condition>
LIMIT <limit_number>
  • 机读(MySQL读取顺序)
    • 《MySQL高级知识-查询与索引优化分析》 机读顺序
FROM
     <left_table>
ON <join_condition>
<join_type> JOIN <right_table> 
WHERE 
    <where_condition>
GROUP BY
    <group_by_list>
HAVING
    <having_condition>
SELECT DISTINCT
    <select_list>
ORDER BY
    <order_by_condition>
LIMIT <limit_number>
  • 总结-SQL解析顺序
    • 《MySQL高级知识-查询与索引优化分析》 SQL解析
SQL JOINs
  • 《MySQL高级知识-查询与索引优化分析》 七种JOIN图解
  • 实验:
-- 建表和数据SQL
CREATE TABLE `tbl_dept` (
 `id` INT(11) NOT NULL AUTO_INCREMENT,
 `deptName` VARCHAR(30) DEFAULT NULL,
 `locAdd` VARCHAR(40) DEFAULT NULL,
 PRIMARY KEY (`id`)
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
 
CREATE TABLE `tbl_emp` (
 `id` INT(11) NOT NULL AUTO_INCREMENT,
 `name` VARCHAR(20) DEFAULT NULL,
 `deptId` INT(11) DEFAULT NULL,
 PRIMARY KEY (`id`),
 KEY `fk_dept_id` (`deptId`)
 #CONSTRAINT `fk_dept_id` FOREIGN KEY (`deptId`) REFERENCES `tbl_dept` (`id`)
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
 
 
 
INSERT INTO tbl_dept(deptName,locAdd) VALUES('RD',11);
INSERT INTO tbl_dept(deptName,locAdd) VALUES('HR',12);
INSERT INTO tbl_dept(deptName,locAdd) VALUES('MK',13);
INSERT INTO tbl_dept(deptName,locAdd) VALUES('MIS',14);
  • 练习
    • 1、A、B两表共有
      • select * from tbl_emp a inner join tbl_dept b on a.deptId = b.id;
    • 2、A、B两表共有+A的独有
      • select * from tbl_emp a left join tbl_dept b on a.deptId = b.id
    • 3、A、B两表共有+B的独有
      • select * from tbl_emp a right join tbl_dept b on a.deptId = b.id;
    • 4、A的独有
      • select * from tbl_emp a left join tbl_dept b on a.deptId = b.id where b.id is null;
    • 5、B的独有
      • select * from tbl_emp a right join tbl_dept b on a.deptId = b.id where a.deptId is null;
    • 6、AB全有
      • MySQL Full Join的实现 因为MySQL不支持FULL JOIN,下面是替代方法
      • left join + union(可去除重复数据)+ right join
      • 实现如下面代码
    • 7、A的独有 + B的独有
-- 6、AB全有
SELECT *
FROM tbl_emp a LEFT JOIN tbl_dept b ON a.deptId = b.id
UNION 
SELECT *
FROM tbl_emp a RIGHT JOIN tbl_dept b ON a.deptId = b.id;


-- 7、A的独有 + B的独有
SELECT *
FROM tbl_emp a LEFT JOIN tbl_dept b ON a.deptId = b.id
WHERE b.id IS NULL
UNION
SELECT *
FROM tbl_emp a RIGHT JOIN tbl_dept b ON a.deptId = b.id
WHERE a.`deptId` IS NULL;

索引简介

什么是索引
  • MySQL官方对索引的定义为:索引(Index)是帮助MySQL高效获取数据的数据结构
    • 可以得到索引的本质:索引是数据结构
    • 索引的目的在于提高查询效率,可以类比字典,
    • 如果要查“mysql”这个单词,我们肯定需要定位到m字母,然后从下往下找到y字母,再找到剩下的sql。
    • 如果没有索引,那么你可能需要a—-z,如果我想找到Java开头的单词呢?或者Oracle开头的单词呢?
    • 是不是觉得如果没有索引,这个事情根本无法完成?
  • 你可以简单理解为“排好序的快速查找结构”
    • 在数据之外,数据库系统还维护着满足特定查找算法的数据结构,这些数据结构以某种方式引用(指向)数据,
      这样就可以在这些数据结构上实现高级查找算法。这种数据结构,就是索引。
    • 下图就是一种可能的索引方式示例:
      • 《MySQL高级知识-查询与索引优化分析》
    • 为了加快Col2的查找,可以维护一个右边所示的二叉查找树,每个节点分别包含索引键值和一个指向对应数据记录物理地址的指针,这样就可以运用二叉查找在一定的复杂度内获取到相应数据,从而快速的检索出符合条件的记录。
  • 一般来说索引本身也很大,不可能全部存储在内存中,因此索引往往以索引文件的形式存储的磁盘上
  • 我们平常所说的索引,如果没有特别指明,都是指B+树结构组织的索引。其中聚集索引,次要索引,覆盖索引,
    复合索引,前缀索引,唯一索引默认都是使用B+树索引,统称索引
    。当然,除了B+树这种类型的索引之外,还有哈稀索引(hash index)等。
索引的优势
  • 类似大学图书馆建书目索引,提高数据检索的效率,降低数据库的IO成本
  • 通过索引列对数据进行排序,降低数据排序的成本,降低了CPU的消耗
索引的劣势
  • 实际上索引也是一张表,该表保存了主键与索引字段,并指向实体表的记录,所以索引列也是要占用空间的
  • 虽然索引大大提高了查询速度,同时却会降低更新表的速度,如对表进行INSERT、UPDATE和DELETE。因为更新表时,MySQL不仅要保存数据,还要保存一下索引文件每次更新添加了索引列的字段,都会调整因为更新所带来的键值变化后的索引信息
  • 索引只是提高效率的一个因素,如果你的MySQL有大数据量的表,就需要花时间研究建立最优秀的索引,或优化查询语句
MySQL索引分类
  • 单值索引
    • 即一个索引只包含单个列,一个表可以有多个单列索引
  • 唯一索引
    • 索引列的值必须唯一,但允许有空值
  • 复合索引
    • 即一个索包含多个列
  • 基本语法
    • 创建,两种方式
      • CREATE [UNIQUE ] INDEX indexName ON mytable(columnname(length));
        • 如果是CHAR,VARCHAR类型,length 可以小于字段实际长度;
        • 如果是 BLOB 和 TEXT 类型,必须指定 length。
      • ALTER mytable ADD [UNIQUE ] INDEX [indexName] ON (columnname(length))
    • 删除
      • DROP INDEX [indexName] ON mytable;
    • 查看
      • SHOW INDEX FROM table_name\G
    • 使用Alter 命令
      • 有四种方式来添加数据表的索引:
        • ALTER TABLE tbl_name ADD PRIMARY KEY (column_list)
          • 该语句添加一个主键,这意味着索引值必须是唯一的,且不能为NULL。
        • ALTER TABLE tbl_name ADD UNIQUE index_name (column_list)
          • 这条语句创建索引的值必须是唯一的(除了NULL外,NULL可能会出现多次)。
        • ALTER TABLE tbl_name ADD INDEX index_name (column_list)
          • 添加普通索引,索引值可出现多次。
        • ALTER TABLE tbl_name ADD FULLTEXT index_name (column_list)
          • 该语句指定了索引为 FULLTEXT ,用于全文索引。
MySQL索引结构
  • BTree索引
    • 检索原理
      • 《MySQL高级知识-查询与索引优化分析》
      • 【初始化介绍】
        • 一颗b+树,浅蓝色的块我们称之为一个磁盘块,可以看到每个磁盘块包含几个数据项深蓝色所示)和指针黄色所示),如磁盘块1包含数据项17和35,包含指针P1、P2、P3。 P1表示小于17的磁盘块,P2表示在17和35之间的磁盘块,P3表示大于35的磁盘块。
        • 真实的数据存在于叶子节点即 3、5、9、10、13、15、28、29、36、60、75、79、90、99。
        • 非叶子节点只不存储真实的数据,只存储指引搜索方向的数据项,如17、35并不真实存在于数据表中。
      • 【查找过程】
        • 如果要查找数据项29,那么首先会把磁盘块1由磁盘加载到内存,此时发生一次IO,在内存中用二分查找确定29在17和35之间,锁定磁盘块1的P2指针,内存时间因为非常短(相比磁盘的IO)可以忽略不计,通过磁盘块1的P2指针的磁盘地址把磁盘块3由磁盘加载到内存,发生第二次IO,29在26和30之间,锁定磁盘块3的P2指针,通过指针加载磁盘块8到内存,发生第三次IO,同时内存中做二分查找找到29,结束查询,总计三次IO。
        • 真实的情况是,3层的b+树可以表示上百万的数据,如果上百万的数据查找只需要三次IO,性能提高将是巨大的,如果没有索引,每个数据项都要发生一次IO,那么总共需要百万次的IO,显然成本非常非常高。
    原文作者:MPPC
    原文地址: https://www.jianshu.com/p/2717d4fbd474
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞