假设有一张订单表 order,主要包含了主键订单编码 order_no、订单状态 status、提交时间 create_time 等列,并且创建了 status 列索引和 create_time 列索引。此时通过创建时间降序获取状态为 1 的订单编码,以下是具体实现代码:
select order_no from order where status =1 order by create_time desc
你知道其中的问题所在吗?我们又该如何优化?
实践
1. 造数据
创建表order01,主键索引,status,create_time 索引
CREATE TABLE `order01` (
`oder_no` bigint(0) NOT NULL AUTO_INCREMENT,
`status` bigint(0) NULL DEFAULT NULL,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL,
`create_time` datetime(0) NULL DEFAULT NULL,
PRIMARY KEY (`oder_no`) USING BTREE,
INDEX `idx_status`(`status`) USING BTREE,
INDEX `idx_create_time`(`create_time`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic;
创建表order02,主键索引,status和create_time 联合索引
CREATE TABLE `order02` (
`id` bigint(0) NOT NULL AUTO_INCREMENT,
`status` bigint(0) NULL DEFAULT NULL,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL,
`create_time` datetime(0) NULL DEFAULT NULL,
PRIMARY KEY (`id`) USING BTREE,
INDEX `idx_status_create_time`(`status`, `create_time`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic;
随机插入10000条数据,方便测试
DROP PROCEDURE IF EXISTS proc_initData;--如果存在此存储过程则删掉
DELIMITER $
CREATE PROCEDURE proc_initData()
BEGIN
DECLARE i INT DEFAULT 1;
WHILE i<=10000 DO
insert into order01(STATUS,name,create_time) VALUES(RAND()*10000000,RAND()*10000000,now());
SET i = i+1;
END WHILE;
END $
CALL proc_initData();
执行完,我手动把1000条数据的status值置为1,方便测试。
2. 查看执行计划
EXPLAIN select * from `order01` WHERE STATUS = 1 ORDER BY create_time;
EXPLAIN select * from `order02` WHERE STATUS = 1 ORDER BY create_time;
执行时间对比
3. 结论
status和create_time单独建索引,在查询时只会遍历status索引对数据进行过滤,不会用到create_time列索引,将符合条件的数据返回到server层,在server层对数据通过快排算法进行排序,Extra列会出现filesort;
应该利用索引的有序性,在status和creat_time列建立联合索引,这样根据status过滤后的数据就是按照create_time排好序的,避免在server层排序。