SQL思考题-实践-解题

假设有一张订单表 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;
image.png
EXPLAIN select * from `order02` WHERE STATUS = 1 ORDER BY create_time;
image.png

执行时间对比


image.png

3. 结论

status和create_time单独建索引,在查询时只会遍历status索引对数据进行过滤,不会用到create_time列索引,将符合条件的数据返回到server层,在server层对数据通过快排算法进行排序,Extra列会出现filesort;
应该利用索引的有序性,在status和creat_time列建立联合索引,这样根据status过滤后的数据就是按照create_time排好序的,避免在server层排序。

你可能感兴趣的:(SQL思考题-实践-解题)