美文网首页
SQL思考题-实践-解题

SQL思考题-实践-解题

作者: lconcise | 来源:发表于2022-04-22 22:36 被阅读0次

    假设有一张订单表 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思考题-实践-解题

          本文链接:https://www.haomeiwen.com/subject/vszgertx.html