位置: 编程技术 - 正文

MySQL多表链接查询核心优化(mysql多表内连接查询)

编辑:rootadmin

推荐整理分享MySQL多表链接查询核心优化(mysql多表内连接查询),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:mysql多表链接查询主要有哪三种类型,mysql多表链接查询主要有哪三种类型,mysql多表联查有几种方法,mysql中进行多表联查方式,mysql多表联查有几种方法,mysql多表连接查询方式,mysql多表链接查询主要有哪三种类型,mysql多表链接查询,内容如对您有帮助,希望把文章链接给更多的朋友!

概述

在一般的项目开发中,对数据表的多表查询是必不可少的。而对于存在大量数据量的情况时(例如百万级数据量),我们就需要从数据库的各个方面来进行优化,本文就先从多表查询开始。其他优化操作,后续另外更新,敬请关注。

数据背景

现假设有一个中学学校,学校中的年级有一年级、二年级、三年级,每个年级有两个班级。分别为、、、、、.

现在我们要为这个学校建立一个考试成绩统计系统。为此,我们对数据库的设计画了如下ER图:

根据ER图,我们设计了数据表,结构如下: class 班级表:

+------------+---------+------+-----+---------+----------------+| Field | Type | Null | Key | Default | Extra |+------------+---------+------+-----+---------+----------------+| id | int() | NO | PRI | NULL | auto_increment || class_name | int() | NO | | NULL | || master_id | int() | YES | | NULL | || is_key | int() | NO | | NULL | |+------------+---------+------+-----+---------+----------------+

student 学生表:

+------------+-------------+------+-----+---------+----------------+| Field | Type | Null | Key | Default | Extra |+------------+-------------+------+-----+---------+----------------+| id | int() | NO | PRI | NULL | auto_increment || school_id | int() | NO | | NULL | || name | varchar() | NO | | NULL | || sex | int() | NO | | NULL | || age | int() | NO | | NULL | || class_name | int() | NO | | NULL | |+------------+-------------+------+-----+---------+----------------+

course 课程表:

+--------------+-------------+------+-----+---------+----------------+| Field | Type | Null | Key | Default | Extra |+--------------+-------------+------+-----+---------+----------------+| id | int() | NO | PRI | NULL | auto_increment || course_name | varchar() | NO | | NULL | || grade | int() | NO | | NULL | || president_id | int() | YES | | NULL | || is_neces | int() | NO | | NULL | || credit | int() | NO | | NULL | || class_name | int() | YES | | NULL | |+--------------+-------------+------+-----+---------+----------------+

score 成绩表:

+-----------+---------+------+-----+---------+----------------+| Field | Type | Null | Key | Default | Extra |+-----------+---------+------+-----+---------+----------------+| id | int() | NO | PRI | NULL | auto_increment || course_id | int() | NO | | NULL | || school_id | int() | NO | | NULL | || score | int() | YES | | NULL | |+-----------+---------+------+-----+---------+----------------+

注:关于本文的数据库数据大家可以在文章最下方的相关下载中获取。资源链接中有两个版本的数据库,school.sql为初始数据库,school_2.sql为优化后的数据库。

连接(JOIN)简介

内连(INNER JOIN)

INNER JOIN 关键字在表中存在至少一个匹配时返回行。

我们也用下面的交集维恩图来描述内连操作:上面的维恩图只是表达了一个有限制情况(即存在JOIN ON),而对于没有约束的情况下,其实就是一个笛卡尔积运算。

*注:**INNER JOIN 与 JOIN 是相同的。一般情况下,在SQL语句中可以省略*INNER关键字。

左连接(LEFT JOIN)

LEFT JOIN 关键字从左表(table1)返回所有的行,即使右表(table2)中没有匹配。如果右表中没有匹配,则结果为 NULL。

使用维恩图描述内连操作:对于上面结果为 NULL的这一条,通过对实际测试的数据表进行操作,得到如下的测试结果:

+------------+-------+| class_name | name |+------------+-------+| | NULL || | Bob || | Alice |+------------+-------+

右连接(RIGHT JOIN)

RIGHT JOIN 关键字从右表(table2)返回所有的行,即使左表(table1)中没有匹配。如果左表中没有匹配,则结果为 NULL。注:右连接可以理解成左连接的对称互补,详细说明可参见左连接。

全连(FULL JOIN)

FULL OUTER JOIN 关键字只要左表(table1)和右表(table2)其中一个表中存在匹配,则返回行.

FULL OUTER JOIN 关键字结合了 LEFT JOIN 和 RIGHT JOIN 的结果。

联合(UNION)

UNION 操作符用于合并两个或多个 SELECT 语句的结果集。

请注意,UNION 内部的每个 SELECT 语句必须拥有相同数量的列。列也必须拥有相似的数据类型。同时,每个 SELECT 语句中的列的顺序必须相同。

MySQL的JOIN实现原理

在MySQL 中,只有一种Join 算法,就是大名鼎鼎的Nested Loop Join,他没有其他很多数据库所提供的Hash Join,也没有Sort Merge Join。顾名思义,Nested Loop Join 实际上就是通过驱动表的结果集作为循环基础数据,然后一条一条的通过该结果集中的数据作为过滤条件到下一个表中查询数据,然后合并结果。如果还有第三个参与Join,则再通过前两个表的Join 结果集作为循环基础数据,再一次通过循环查询条件到第三个表中查询数据,如此往复。                                        ? 《MySQL性能调优与架构设计》

多表查询实战

查询各个班级的班长姓名

优化分析

对于这个多表的查询使用where是可以很好地完成查询,而查询的结果从表面上看,完全没什么问题,如下:

+------------+---------+| class_name | name |+------------+---------+| | William || | Peter || | Judy || | Polly || | Grace || | Sunny |+------------+---------+

可是,由于我们使用的是where,这个与内连接在有条件限制的情况下是一样的,其维恩图也可以一并参考。可是,如果现在我们假设,有一个新的班级,或是这个的班级暂时还没有班长。这个时候通过where就无法完成查询了。上面的结果中就已经很好地给出解释。

这个时候,我们就需要通过外连接中的左连接(如果采用右连接,那么相应的表位置也要进行替换)来进行查询了。在左连的查询中,因为是包含了”左表“的全部行,所以对于未选出班长的来说,这个很有必要。采用左连操作的结果如下:

+------------+---------+| class_name | name |+------------+---------+| | William || | Peter || | Judy || | Polly || | Grace || | Sunny || | NULL |+------------+---------+

SQL展示

朴素的WHERE

MySQL多表链接查询核心优化(mysql多表内连接查询)

INNER JOIN

LEAF JOIN

RIGHT JOIN

利用 EXPLAIN 检查优化器

通过EXPLAIN我们分别检查上面WHERE语句和LEFT JOIN的优化过程。结果如下:

WHERE

+----+-------------+-------+------+---------------+------+---------+------+------+--------------------------------+| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |+----+-------------+-------+------+---------------+------+---------+------+------+--------------------------------+| 1 | SIMPLE | cl | ALL | NULL | NULL | NULL | NULL | 7 | || 1 | SIMPLE | st | ALL | NULL | NULL | NULL | NULL | | Using where; Using join buffer |+----+-------------+-------+------+---------------+------+---------+------+------+--------------------------------+

LEFT JOIN

+----+-------------+-------+------+---------------+------+---------+------+------+-------+| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |+----+-------------+-------+------+---------------+------+---------+------+------+-------+| 1 | SIMPLE | cl | ALL | NULL | NULL | NULL | NULL | 7 | || 1 | SIMPLE | st | ALL | NULL | NULL | NULL | NULL | | |+----+-------------+-------+------+---------------+------+---------+------+------+-------+

对于上面的两个结果,我们可以看到有一个很明显的区别在于Extra。

Using where说明进行了where的过滤操作,Using join buffer说明进行join缓存。

从上面的结果中,还可以看到每种情况的两种查询操作都是经过了全表扫描。而这对于大量数据而言是很不利的。

现在,我们可以为被驱动表的join字段添加索引,再对其进行EXPLAIN检查。

添加索引

通过EXPLAIN我们分别检查上面WHERE语句和LEFT JOIN的优化过程。结果如下:

WHERE

+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+| 1 | SIMPLE | cl | ALL | NULL | NULL | NULL | NULL | 7 | || 1 | SIMPLE | st | ref | index_school_id | index_school_id | 4 | school.cl.master_id | 1 | |+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+

LEFT JOIN

+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+| 1 | SIMPLE | cl | ALL | NULL | NULL | NULL | NULL | 7 | || 1 | SIMPLE | st | ref | index_school_id | index_school_id | 4 | school.cl.master_id | 1 | |+----+-------------+-------+------+-----------------+-----------------+---------+---------------------+------+-------+

现在,可以很明显地看出rows列的数值,在被驱动表处都是1,这大降低了查询的复杂度。而且对于type列,也从一开始的ALL变成了现在的ref。还有一些其他的列也被修改了。

查询番外

根据学号查询一个学生的成绩单

WHERE 查询

JOIN 查询

结果

+----+-------------+-------+--------+---------------------------------------+--------------------+---------+---------------------+------+-------+| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |+----+-------------+-------+--------+---------------------------------------+--------------------+---------+---------------------+------+-------+| 1 | SIMPLE | st | ref | index_school_id | index_school_id | 4 | const | 1 | || 1 | SIMPLE | sc | ref | index_school_id_sc,index_course_id_sc | index_school_id_sc | 4 | const | 3 | || 1 | SIMPLE | co | eq_ref | PRIMARY | PRIMARY | 4 | school.sc.course_id | 1 | |+----+-------------+-------+--------+---------------------------------------+--------------------+---------+---------------------+------+-------+

优化总结

对于要求全面的结果时,我们需要使用连接操作(LEFT JOIN / RIGHT JOIN / FULL JOIN); 不要以为使用MySQL的一些连接操作对查询有多么大的改善,核心是索引; 对被驱动表的join字段添加索引;

SQL语句表

创建数据库

创建数据表

学生表

班级表

课程表

成绩表

导入导出

索引操作

查询实战

查询所有课程名称

查询一个学生全部课程

统计每个班级有多少学生

根据学号查询一个学生的成绩单

查询各个班级的班长姓名

其他查询

原文链接:

利用rpm安装mysql 5.6版本详解 前言其实之前使用yum安装MySQL确实很方便,但是默认安装的myql5.0版本的,不支持utf8mb4(utf8mb4扩展到一个字符最多能有4节,所以能支持更多的字符集,比

MySQL数据库的一次死锁实例分析 1、故事起因于年月日的一个生产bug。业务场景是:归档一个表里边的数据到历史表里边,同是删除主表记录。2、背景场景简化如下(数据库引擎I

Mysql事项,视图,函数,触发器命令(详解) 事项开启和使用//修改表的引擎altertableaengine=myisam;//开启事务begin;//关闭自动提交setautocommit=0;//扣updatebanksetmoney=money-wherebid=1;//回滚,begin开始的所有

标签: mysql多表内连接查询

本文链接地址:https://www.jiuchutong.com/biancheng/349269.html 转载请保留说明!

上一篇:Linux下彻底卸载mysql详解(linux彻底卸载软件)

下一篇:利用rpm安装mysql 5.6版本详解(利用rpm安装软件包时,应使用命令选项)

  • 医疗机构销售药品需要销售记录吗
  • 申报个税时怎么获取个税专项附加扣除
  • 抄税是什么意思谁做的事情
  • 快递公司快递人员工资记入什么会计科目
  • 高速公路通行费发票怎么开
  • 购买旧固定资产的增值税进项税
  • 分步法和分批法思维导图
  • 什么情况下要安起搏器
  • 代扣公积金会计怎么做账
  • 进项税额抵减欠税后附加税可以抵减吗
  • 未收到发票暂估入库的进项税额可以暂估进原料成本吗
  • 房地产企业预缴税费
  • 企业所得税汇算清缴退税分录
  • 没有运输经营范围能不能开运输发票
  • 季报每个月需要反写吗
  • 简易征收税额航天信息服务费可以抵扣吗?
  • 印花税一定要贴花吗
  • 开专用发票的地址有错会有影响吗?
  • 沥青混凝土增值税风险
  • 地方附加税
  • 建筑业增值税税率2022
  • 税务行业软件
  • 在下列哪种情况下
  • 对公账户分为一般户和基本户
  • 2021年电子税务局印花税怎么申报
  • 办公室零食知乎
  • 车间办公费计入什么会计科目
  • 机票的进项
  • 午餐补贴多少钱
  • 存货质押融资的银行
  • 个税申报月份有误如何调整?
  • 合作社人工工资账务处理
  • 价外费用含义
  • 取得进账发票没发票
  • 小规模纳税人确认收入时要确认税吗
  • wordpress网站打开很慢
  • 个人二手车转让协议书免费
  • 注销企业基本户需要先注销一般户吗
  • PHP:preg_split()的用法_PCRE正则函数
  • 为什么把收入进行舞弊假定
  • 结转本月发生的费用
  • 没有独立显卡没有核显能开机吗
  • PHP:mdecrypt_generic()的用法_Mcrypt函数
  • python网络爬虫技术
  • php如何生成html
  • css元素居中的几种方法
  • 保险费支出是什么意思
  • 一般人转小规模政策文号
  • idea快速生成lambda
  • 新会计准则下的资产负债表
  • 投资款未备注
  • 公章变更需要到哪个部门
  • 电子税务局怎么注册
  • 施工企业结算单能不能入账
  • 购买其他权益工具投资公允大于我支付的价值
  • 工资算营业成本还是管理费用
  • 缴纳社保公积金的会计分录
  • 已经发出的商品替换零部件进成本还是费用
  • 红字发票抵扣联和发票联要给对方吗
  • 企业为何
  • 个人独资企业与一人公司的区别与联系
  • 税控盘全额抵扣怎么做分录
  • 商业承兑汇票到期不兑付怎么办
  • 无生产经营收入可以评为a吗
  • 会计档案步骤
  • 赊销是什么意思 视频
  • mysql的增删改查命令
  • window8更新
  • usrmlnka.exe - usrmlnka进程是什么意思
  • 如何更改win7
  • VirtualBOX给CentOS建共享文件夹的方法
  • linux tar -czvf
  • linux012
  • js require()
  • cocos2dx入门
  • python2.7.11
  • go语言websocket
  • js实现手风琴
  • 国家税务总局党建工作局
  • 必须抓好什么治理能力和治理体系
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设