优化Using join buffer (Block Nested Loop)

文章通过四个场景探讨了SQL查询优化中的关键因素,包括关联字段是否添加索引、右表数据量大小、字段类型一致性以及是否使用联合索引。在每个场景中,都展示了如何通过调整这些因素来提高查询性能,例如为关联字段添加索引、增加右表数据、保持字段编码一致性和创建联合索引。
摘要由CSDN通过智能技术生成

场景1:检查表之间的关联字段有没有添加索引

若右边的表的关联的字段没有索引,整体sql查询也不会走索引

表结构

学生表:有10万数据

CREATE TABLE `student` (
  `id` bigint(20) NOT NULL COMMENT '学生id',
  `class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `age` int(2) DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `class` (`class`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

学生评级表:有3条的数据

CREATE TABLE `student_grade` (
  `id` bigint(20) NOT NULL COMMENT '序号',
  `student_id` bigint(20) DEFAULT NULL COMMENT '学生id',
  `student_class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `create_time` datetime DEFAULT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`),
  KEY `create_time` (`create_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

执行sql

EXPLAIN SELECT
	s.id
FROM
	student s
	LEFT JOIN student_grade g ON s.id = g.student_id 
	AND s.class = g.student_class

结果分析

仔细分析发现student_grade的student_id、student_class字段没有添加索引,添加索引即可

场景2:检查表之间的右边的表的数据量

若右边的表的数据量很少,也不会走索引

表结构

学生表:有10万数据

CREATE TABLE `student` (
  `id` bigint(20) NOT NULL COMMENT '学生id',
  `class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `age` int(2) DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `class` (`class`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

学生评级表:有3条的数据

CREATE TABLE `student_grade` (
  `id` bigint(20) NOT NULL COMMENT '序号',
  `student_id` bigint(20) DEFAULT NULL COMMENT '学生id',
  `student_class` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '学生班级',
  `create_time` datetime DEFAULT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`),
  KEY `student_id` (`student_id`),
  KEY `student_class` (`student_class`),
  KEY `create_time` (`create_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;

执行sql

EXPLAIN SELECT
	s.id
FROM
	student s
	LEFT JOIN student_grade g ON s.id = g.student_id 
	AND s.class = g.student_class

执行结果

在这里插入图片描述

结果分析

仔细检查发现是s.class = g.student_class中关联字段编码一样,然而student_grade的数据量太少,于是在student_grade表增加一条记录,结果走索引了
在这里插入图片描述

场景3:检查表之间的关联字段类型是否一致

表结构

学生表:有10万数据

CREATE TABLE `student` (
  `id` bigint(20) NOT NULL COMMENT '学生id',
  `class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `age` int(2) DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `class` (`class`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

学生评级表:有3条以上的数据

CREATE TABLE `student_grade` (
  `id` bigint(20) NOT NULL COMMENT '序号',
  `student_id` bigint(20) DEFAULT NULL COMMENT '学生id',
  `student_class` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '学生班级',
  `create_time` datetime DEFAULT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`),
  KEY `student_id` (`student_id`),
  KEY `student_class` (`student_class`),
  KEY `create_time` (`create_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;

执行sql

EXPLAIN SELECT
	s.id
FROM
	student s
	LEFT JOIN student_grade g ON s.id = g.student_id 
	AND s.class = g.student_class

执行结果

在这里插入图片描述

结果分析

仔细检查发现是s.class = g.student_class中关联字段编码不一样,s.class的编码是utf8,g.student_class的编码是utf8mb4
将编码改成一致之后执行sql就走索引了
在这里插入图片描述

场景4:添加联合索引

查询的字段和表之间的关联字段之间不是联合索引

表结构

学生表:有10万数据

CREATE TABLE `student` (
  `id` bigint(20) NOT NULL COMMENT '学生id',
  `class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `age` int(2) DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `class` (`class`),
  KEY `age` (`age`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

学生评级表:有3条以上的数据

CREATE TABLE `student_grade` (
  `id` bigint(20) NOT NULL COMMENT '序号',
  `student_id` bigint(20) DEFAULT NULL COMMENT '学生id',
  `student_class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `create_time` datetime DEFAULT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`),
  KEY `create_time` (`create_time`),
  KEY `student_class` (`student_class`),
  KEY `student_id` (`student_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

执行sql

EXPLAIN
SELECT
	s.id,s.class,s.age,g.create_time
FROM
	student s
	LEFT JOIN student_grade g ON ( s.id = g.student_id AND s.class = g.student_class )

结果分析

查询student的id,class,age之间的字段没有建立联合索引
在这里插入图片描述

解决方案

加上联合索引,结果发现走索引了

CREATE TABLE `student` (
  `id` bigint(20) NOT NULL COMMENT '学生id',
  `class` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci DEFAULT NULL COMMENT '学生班级',
  `age` int(2) DEFAULT NULL COMMENT '年龄',
  PRIMARY KEY (`id`),
  KEY `id` (`id`,`class`,`age`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

以上结果仅供参考,具体根据实际的sql语句进行优化

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值