索引下推(INDEX CONDITION PUSHDOWN,简称 ICP)是在 MySQL 5.6 针对扫描二级索引的一项优化改进。总的来说是通过把索引过滤条件下推到存储引擎,来减少 MySQL 存储引擎访问基表的次数以及 MySQL 服务层访问存储引擎的次数。ICP 适用于 MYISAM 和 INNODB,本篇的内容只基于 INNODB。
(1)MySQL 服务层:也就是 SERVER 层,用来剖析 SQL 的语法、语义、天生查询筹划、接管从 MySQL 存储引擎层上推的数据进行二次过滤等等。
(2)MySQL 存储引擎层:按照 MySQL 服务层下发的哀求,通过索引或者全表扫描等方式把数据上传到 MySQL 服务层。
(3)MySQL 索引扫描:根据指定索引过滤条件,遍历索引找到索引键对应的主键值后回表过滤剩余过滤条件。
(4)MySQL 索引过滤:通过索引扫描并且基于索引进行二次条件过滤后再回表。
(5)使用索引下推实现
【2】案例
(1)索引下推的使用条件
(1)ICP目标是减少全行记录读取,从而减少IO 操纵,只能用于非聚簇索引。聚簇索引本身包罗的表数据,也就不存在下推一说。
(2)只能用于range、 ref、 eq_ref、ref_or_null访问方法;
(3)where 条件中是用 and 而非 or 的时候。
(4)ICP适用于分区表。
(5)ICP不支持基于假造列上创建的索引,比如说函数索引
(6)ICP不支持引用子查询作为条件。
(7)ICP不支持存储函数作为条件,由于存储引擎无法调用存储函数。
(2)索引下推相干语句
# 查看索引下推是否开启
select @@optimizer_switch
# 开启索引下推
set optimizer_switch="index_condition_pushdown=on";
# 关闭索引下推
set optimizer_switch="index_condition_pushdown=off";
复制代码
进行详细的测试:
CREATE DATABASE icp_demo;
USE icp_demo;
CREATE TABLE employees (
emp_id INT AUTO_INCREMENT PRIMARY KEY,
emp_name VARCHAR(255),
dept_id INT,
salary DECIMAL(10, 2),
hire_date DATE,
INDEX idx_dept_salary (dept_id, salary)
) ENGINE=InnoDB;
复制代码
DELIMITER $$
CREATE PROCEDURE populate_employees()
BEGIN
DECLARE i INT DEFAULT 1;
WHILE i <= 100000 DO
INSERT INTO employees (emp_name, dept_id, salary, hire_date)
VALUES (
CONCAT('Employee_', i),
FLOOR(RAND() * 10),
ROUND(RAND() * 100000, 2),
CURDATE() - INTERVAL FLOOR(RAND() * 3650) DAY
);
SET i = i + 1;
END WHILE;
END$$
DELIMITER ;
复制代码
CALL populate_employees();
复制代码
在执行查询之前,我们可以先查看索引下推是否开启。
SELECT @@optimizer_switch;
复制代码
SET optimizer_switch = 'index_condition_pushdown=on';
复制代码
EXPLAIN SELECT emp_id, emp_name FROM employees WHERE dept_id = 5 AND salary BETWEEN 50000 AND 80000;
复制代码
SET optimizer_switch = 'index_condition_pushdown=off';
复制代码
EXPLAIN SELECT emp_id, emp_name FROM employees WHERE dept_id = 5 AND salary BETWEEN 50000 AND 80000;
复制代码
记得再次打开:
SET optimizer_switch = 'index_condition_pushdown=on';