数据库调优-SQL语句优化

2. SQL语句优化

 

sql

复制代码

# 请问这两条SQL语句有什么区别呢?你来猜一猜那条SQL语句执行查询效果更好! select id from sys_goods where goods_name='华为 HUAWEI 麦芒7 魅海蓝 6G+64G 全网通'; ​ select id from sys_goods where goods_id=1496732598527979521;

2.1. 查看SQL执行计划【explain】

MySQL 提供了一个 EXPLAIN 命令, 它可以对 SELECT 语句的执行计划进行分析, 并输出 SELECT 执行的 详细信息, 以供开发人员针对性优化。使用explain这个命令来查看一个这些SQL语句的执行计划,查看 该SQL语句有没有使用上了索引,有没有做全表扫描,这都可以通过explain命令来查看。

可以通过explain命令深入了解MySQL的基于开销的优化器,还可以获得很多可能被优化器考虑到的访 问策略的细节,以及当运行SQL语句时哪种策略预计会被优化器采用。

EXPLAIN 命令用法十分简单, 在 SELECT 语句前加上 explain 就可以了, 例如:

在MySQL中可以使用explain 查看SQL执行计划,用法:

 

sql

复制代码

EXPLAIN SELECT * FROM t_user;

image-20220729134757183.png

  1. id:SELECT识别符,这是SELECT查询序列号。
  2. select_type(重要):表示单位查询的查询类型,比如:普通查询、联合查询(union、union all)、子查询等复杂查询。
  3. table:表示查询的表。
  4. partitions:使用的哪些分区(对于非分区表值为null)。
  5. type(重要)表示表的连接类型。
  6. possible_keys:此次查询中可能选用的索引,一个或多个
  7. key:查询真正使用到的索引
  8. key_len:显示MySQL决定使用的索引长度。
  9. ref:哪个字段或常数与 key 一起被使用
  10. rows:显示此查询一共扫描了多少行,这个是一个估计值。不精确值
  11. filtered: 表示此查询条件所过滤的数据的百分比
  12. Extra(重要):额外信息

哈哈,看到这么多的参数,可能,大部分人都直接放弃了!!!

xdm,别怕,重点的参数,我已经标出来了,也就3个,那我们就针对这3个参数进行详细说明吧。

select_type:查询类型

type:连接类型

Extra:额外信息

哈哈,开干!!!

image.png

2.2. 关键结果说明

2.2.1 select_type

单位查询的查询类型,比如:普通查询、联合查询(union、union all)、子查询等复杂查询。

有以下几种值:

  • simple: 普通查询,表示不需要union操作或者不包含子查询的简单select查询。有连接查询时,外层的查询为simple。

  • primary: 查询的主要部分, 一个需要union操作或者含有子查询的select,位于最外层的单位查询的select_type即为primary。

  • union:连接查询 ,union连接的两个select查询,第一个查询是dervied派生表,除了第一个表外,第二个以后的表select_type都是union

    • derived在FROM列表中包含的子查询被标记为DERIVED(衍生),MySQL会递归执行这些子查询,把结果放在临时表中
    • union若第二个SELECT出现在UNION之后,则被标记为UNION:若UNION包含在FROM子句的子查询中,外层SELECT将被标记为:DERIVED
    • union result 从UNION表获取结果的SELECT
  • dependent union: 依赖连接查询 ,与union一样,出现在union 或union all语句中,但是这个 查询要受到外部查询的影响
  • subquery: 子查询 ,除了from字句中包含的子查询外,其他地方出现的子查询都可能是subquery
  • dependent subquery: 依赖子查询, 与dependent union类似,表示这个subquery的查询要受到外部表查询的影响
  • derived:派生表, from字句中出现的子查询,也叫做派生表,其他数据库中可能叫做内联视图或嵌套select
01-simple简单查询
 

csharp

复制代码

-- simple: 表示不需要union操作或者不包含子查询的简单select查询。有连接查询时,外层的查询为simple。 ​ explain select * from t_user;

image-20220729134757183.png

02-union连接查询
 

sql

复制代码

-- union: union连接的两个select查询,第一个查询是dervied派生表,除了第一个表外,第二个以后的表select_type都是union -- 语句 1 explain select * from t_user a union select * from t_user b; ​ -- UNION 若第二个SELECT出现在UNION之后,则被标记为UNION: -- 若UNION包含在FROM子句的子查询中,外层SELECT将被标记为:DERIVED -- UNION RESULT 从UNION表获取结果的SELECT -- 语句 2 explain select * from ( select * from t_user a union select * from t_user b) c; -- DERIVED 在FROM列表中包含的子查询被标记为DERIVED(衍生),MySQL会递归执行这些子查询,把结果放在临时表中

image-20220729135602788.png

image-20220729135627313.png

 

sql

复制代码

-- dependent union:与union一样,出现在union 或union all语句中,但是这个查询要受到外部查询的影响 ​ explain select * from t_user a where a.id in (select id from t_user b union select id from t_user c);

image-20220729135929456.png

03-subquery子查询
 

csharp

复制代码

-- subquery 除了from字句中包含的子查询外,其他地方出现的子查询都可能是subquery ​ explain select (select id from t_user where id=1) from t_user;

image-20220729140106297.png

 

css

复制代码

-- dependent subquery 与dependent union类似,表示这个subquery的查询要受到外部表查询的影响 ​ explain select (select id from t_user a where a.id=b.id) from t_user b;

image-20220729140208880.png

2.2.2 type

显示的是单位查询的 查询类型 或者理解为 访问类型 ,访问性能依次从好到差:

 

sql

复制代码

system const eq_ref ref fulltext ref_or_null unique_subquery index_subquery range index_merge index ALL:查询效率很低,而且耗CPU,在数据量大的情况下才会出现

  • system :表中 只有一行数据或者是空表 。等于系统表,这是const类型的特列,平时不会出现,可以忽略不计

  • const(重要) :使用 唯一索引或者主键 ,返回记录一定是 1 行记录的等值where条件时,通常type是const。其他数据库也叫做唯一索引扫描。

  • eq_ref(重要) : 唯一性索 引扫描,对于每个索引键,表中只有一条记录与之匹配。常见于主键或唯一索引扫描

  • ref(重要) : 非唯一性索 引扫描,返回匹配某个单独值的所有行,本质上也是一种索引访问,它返回所有匹配某个单独值的行,然而,它可能会找到多个符合条件的行,所以他应该属于查找和扫描的混合体。

    • 组合索引
    • 非唯一索引
  • fulltext :全文索引检索,要注意,全文索引的优先级很高,若全文索引和普通索引同时存在时,mysql不管代价,优先选择使用全文索引

  • ref_or_null :与ref方法类似,只是增加了null值的比较。实际用的不多。

  • unique_subquery :用于where中的in形式子查询,子查询返回不重复值唯一值

  • index_subquery :用于in形式子查询使用到了辅助索引或者in常数列表,子查询可能返回重复值,可以使用索引将子查询去重。

  • range(重要) : 索引范围扫描 ,常见于使用>,<,is null,between ,in ,like等运算符的查询中。

  • index_merge :表示查询使用了两个以上的索引,最后取交集或者并集,常见and ,or的条件使用了不同的索引,官方排序这个在ref_or_null之后,但是实际上由于要读取所个索引,性能可能大部分时间都不如range

  • index(重要) :select结果列中使用到了索引,type会显示为index。 全部索引扫描 ,把索引从头到尾扫一遍,常见于使用索引列就可以处理不需要读取数据文件的查询、可以使用索引排序或者分组的查询。

  • all(重要) :这个就是全表扫描数据文件,然后再 在server层进行过滤 返回符合要求的记录。

  • 除了all之外,其他的type都可以使用到索引
  • 除了index_merge之外,其他的type只可以用到一个索引
  • 最少要使用到range级别

详解:

01-const:
  • 使用 唯一索引或者主键 ,返回记录一定是 1 行记录的等值where条件时,通常type是const。其他数据库也叫做唯一索引扫描。
 

csharp

复制代码

explain select * from t_user where id=1;

image-20220729140834173.png

02-eq_ref:
  • 唯一性索引扫描,对于每个索引键,表中只有一条记录与之匹配。常见于主键或唯一索引扫描
 

css

复制代码

explain select * from t_user a left join t_user b on a.id=b.id;

image-20220729140949176.png

03-ref:
  • 非唯一性索引扫描,返回匹配某个单独值的所有行,本质上也是一种索引访问,它返回所有匹配某个单独值的行,然而,它可能会找到多个符合条件的行,所以他应该属于查找和扫描的混合体。

组合索引

 

csharp

复制代码

-- ref 组合索引 explain select * from t_user where username="llsydn"; ​ explain select * from t_user a left join t_user b on a.username=b.username;

image-20220729141413888.png

image-20220729141519170.png

非唯一索引

 

csharp

复制代码

-- ref 非唯一索引 explain select * from t_user where account='llsydn';

image-20220729141939897.png

04-range:
  • 索引范围扫描 ,常见于使用>,<,is null,between ,in ,like等运算符的查询中。
 

sql

复制代码

explain select * from t_user where account like 'lls%';

image-20220729142332201.png

05-index:
  • select结果列中使用到了索引,type会显示为index。 全部索引扫描 ,把索引从头到尾扫一遍,常见于使用索引列就可以处理不需要读取数据文件的查询、可以使用索引排序或者分组的查询。
 

csharp

复制代码

explain select account from t_user;

image-20220729142532347.png

06-all:
  • 这个就是全表扫描数据文件,然后再 在server层进行过滤 返回符合要求的记录。
 

csharp

复制代码

1 explain select * from t_user;

image-20220729142633514.png

2.2.3 Extra

这个列包含不适合在其他列中显示的,但十分重要的额外的信息,这个列可以显示的信息非常多,有几十种。解释几个经常遇到的

01-Using filesort
  • 使用了文件排序,说明mysql会对数据使用一个外部的索引排序,而不是按照表内的索引顺序进行读取。MySQL中无法利用索引完成的排序操作称为“文件排序”。这种操作需要优化sql。
 

sql

复制代码

-- 无索引,按照文件排序 explain select sort_number from t_menu order by sort_number;

image-20220729144325322.png

 

sql

复制代码

-- 有索引,不按照文件排序 explain select sort_number from t_menu order by sort_number;

image-20220729144404825.png

02-using index

表示相应的SELECT查询中使用到了索引,避免访问表的数据行,这种查询的效率很高!

  • 如果同时出现Using Where ,索引在where之后,用作查询条件
  • 如果没有同时出现Using Where ,索引在where之前,用作查询结果读取
 

vbnet

复制代码

-- 使用where,索引在where之后,用作查询条件 explain select sort_number from t_menu where id>0 order by sort_number;

image-20220729144547441.png

 

vbnet

复制代码

-- 没有使用where,索引在where之前,用作查询结果读取 explain select sort_number from t_menu where id>0 order by sort_number;

image-20220729144931034.png

03-using where
  • 表示Mysql将对storage engine提取的结果进行过滤,过滤条件字段无索引;
 

csharp

复制代码

-- 只有where explain select * from t_menu where id>0;

image-20220729144830097.png

04-Using join buffer
  • 表明使用了连接缓存,比如说在查询的时候,多表join的次数非常多,那么将配置文件中的缓冲区的join buffer调大一些。

image-20220729141519170.png

2.3. 为什么使用索引就能加快查询速度呢?

二叉树数据结构,与搜索算法时间复杂度:

B+Tree

面试常问题目:有没有做过数据库优化呀?我来问你一个特别简单的问题,查询的是有没有加过索引?

为什么要加索引?加上了索引之后速度就会变快吗?请您简单描述一下原因吧?

  • 什么是索引:索引就是事先排好顺序,然后在查询的时候使用二分法等高校的查询算法来进行高效查询。
  • 除了索引查询,肯定一般查询:这两者的差异是数量级的差异。
  • 二分法或索引查询的时间复杂度O(log2N),使用一般查询时间的复杂度是O(n)

举例来说:100w条数据,一般查询,平均 50 万条,比较 50 万次。如果用二分法,只需要不超过 20 次就可以找到数据!

二分法这种查找方式,效率是一般查询的2.5万倍!

哈哈,这里还是直接mark下吧!!!

 下面是配套资料,对于做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!

软件测试面试小程序
被百万人刷爆的软件测试题库!!!谁用谁知道!!!全网最全面试刷题小程序,手机就可以刷题,地铁上公交上,卷起来!

涵盖以下这些面试题板块:

1、软件测试基础理论 ,2、web,app,接口功能测试 ,3、网络 ,4、数据库 ,5、linux 6、web,app,接口自动化 ,7、性能测试 ,8、编程基础,9、hr面试题 10、开放性测试题,11、安全测试,12、计算机基础

​编辑资料获取方式 :xiaobei_upup,添加时备注“csdn alex”

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/9546.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis学习1——redis简介、基础

介绍 redis简介 Redis(Remote Dictonary Server) 是由Salvatore Sanfilippo开发的key-value缓存数据库&#xff0c;基于C语言开发。目前市面上&#xff0c;Redis和MongoDB是当前使用最广泛的NoSQL&#xff0c;而就Redis技术而言&#xff0c;它的性能十分优越&#xff0c;可以…

rngd: Error writing /dev/tpm0

检查数据库时发现messages中一直有rngd报错&#xff0c;rngd一直未配置&#xff0c;直接关闭了 /var/log/messages-20240414:Apr 11 04:59:49 hydb2 rngd: Error writing /dev/tpm0 /var/log/messages-20240414:Apr 12 07:31:39 hydb2 rngd: Error writing /dev/tpm0 /var/log…

shell-for循环语句练习题

1.计算从1到100所有整数的和 [rootlocalhost ~]# vim 1.sh #!/bin/bash sum0 #定义变量sum初始值为0 for i in {1..100} #for循环 i取值从1到100 do sum$[ isum ] #在每次循环中&#xff0c;变量i的值会依次取1到100的整数值。 #sum是一个累加器&#xff0c;初始值…

echarts自定义图例显示名称、数值、占比

先上代码 legend: {orient: vertical,left: 10,top:20,data: data,textStyle: {color: #9FB7D5 // 设置图例文字颜色为白色},// type: plain, // 设置图例类型为普通类型itemWidth: 10, // 设置图例项的宽度itemHeight: 10, // 设置图例项的高度formatter: function(name) {let…

Python专题:十、字典(2)

字典定义x{} get()函数 get&#xff08;参数一&#xff0c;参数二&#xff09; 参数一&#xff1a; 需要查找的关键词 参数二&#xff1a; 如果关键词不存在get返回的默认值 字典的更新 update&#xff08;&#xff09;函数&#xff0c;字典y的元素&#xff0c;去更新字…

学生管理系统初级

根据题目要求生成大纲 总结: 1.在书写时&#xff0c;考虑到了书写时id可是是abc... 类型是String&#xff0c;但在根据id获取集合中元素时 list.get() &#xff0c;get&#xff08;&#xff09;里面是int类型。 2.在书写还有一点功能并不完全&#xff0c; 2.1查找时是打印所有…

十.吊打面试官系列-Tomcat优化-通过压测Tomcat调优实战

前言 上一篇文章我们讲解了一下Tomcat底层的结构和执行原理&#xff0c;我们需要重点去掌握的是Tomcat的高内聚低耦合的设计&#xff0c;以及责任链模式&#xff0c;以及Tomcat NIO编程模式&#xff0c;这些是Tomcat比较核心的点&#xff0c;本篇文章我们将对Tomcat的参数做一…

Java学习【类与对象】

类和对象 开始我们就不讲那些把大象放冰箱需要几步来引入面向对象的例子了&#xff0c;直接上干货。 在Java中&#xff0c;类是对现实世界中某一类事物的抽象描述。它包含了该类事物的属性和方法。属性用于描述事物的状态&#xff0c;而方法则用于描述事物可以做的事情。对象也…

Redis 支持的 Java 客户端都有哪些?

Redis 是一种高性能的键值存储系统&#xff0c;它以其快速、灵活和可扩展的特性而闻名。在 Java 开发中&#xff0c;与 Redis 交互的方式通常是通过使用 Redis 的 Java 客户端。 这些客户端提供了访问 Redis 数据库的接口&#xff0c;使开发人员能够在 Java 应用程序中轻松地使…

电脑设置在哪里打开?Window与Mac双系统操作指南

随着科技的不断发展&#xff0c;电脑已经成为我们日常生活和工作中不可或缺的一部分。然而&#xff0c;对于许多初学者来说&#xff0c;如何找到并熟悉电脑的设置界面可能是一个挑战。特别是对于那些同时使用Windows和Mac双系统的用户来说&#xff0c;更是需要一篇详尽的指南来…

qml 和 c++类的数据交互

1、 新建一个需要交互的C++类 1)添加QObject头文件 2)添加自QObject的继承 3)添加Q_OBJECT宏 4)使用Q_PROPERTY,定义两个交互的属性,并设置读写的方法和变更属性的信号。 5)添加方法、槽函数和变量 2、在main.cpp中添加实例化对象的QML上下文 1)添加需要QML交互的…

在做题中学习(54):点名

LCR 173. 点名 - 力扣&#xff08;LeetCode&#xff09; 此题有不同的几种解法&#xff1a; 解法一&#xff1a;暴力枚举 O(n); 解法二&#xff1a;哈希表 把原数组丢入哈希表&#xff0c;遍历哈希表&#xff0c;看看哪个数值为0即可。 O(n)空间O(n)时间 解法三&…

Redis如何避免数据丢失?——RDB

目录 1. RDB机制的配置 2. fork()函数和写时复制(Copy On Write&#xff09; 什么是Copy On Write 系统fork中使用Copy On Write机制 3. RDB文件结构 RDB文件内容和内容顺序 InfoAuxFields是rdb信息数据 数据库数据 数据 数据存储格式 字符串编码 操作码 4. RDB的2…

【自研网关系列】数字签名和url动态加密

&#x1f308;Yu-Gateway&#xff1a;&#xff1a;基于 Netty 构建的自研 API 网关&#xff0c;采用 Java 原生实现&#xff0c;整合 Nacos 作为注册配置中心。其设计目标是为微服务架构提供高性能、可扩展的统一入口和基础设施&#xff0c;承载请求路由、安全控制、流量治理等…

【Java EE】数据库连接池详解

文章目录 &#x1f38d;数据库连接池&#x1f338;Hikari&#x1f338;Druid &#x1f340;MySQL开发企业规范⭕总结 &#x1f38d;数据库连接池 在上⾯Mybatis的讲解中,我们使⽤了数据库连接池技术,避免频繁的创建连接,销毁连接 下⾯我们来了解下数据库连接池 数据库连接池负…

「ETL实战」搭建数仓,解决多源业务系统关联分析难题(定制化业务)

在大数据分析盛行的今天&#xff0c;关联分析作为数据挖掘和业务洞察的重要手段&#xff0c;受到了极大关注。然而&#xff0c;随着数据量的激增和源业务系统的复杂性增加&#xff0c;关联分析的性能问题逐渐成为了一个不可忽视的挑战。 本文将介绍借助ETL工具&#xff0c;如何…

大数据面试题 —— 数据库

目录 关系型数据库与非关系型数据库的区别数据库三范式MySQL中 drop、delete、truncate的区别MySQL中 char和 varchar 的区别MySQL中inner join、left join、right join以及full join的区别MySQL中 having 和 where 的区别count(*)、count(1)、count(列名)的区别MySQL中视图和表…

Threejs 学习笔记 | 灯光与阴影

文章目录 Threejs 学习笔记 | 灯光与阴影如何让灯光照射在物体上有阴影LightShadow - 阴影类的基类平行光的shadow计算投影属性 - DirectionalLightShadow类平行光的投射相机 聚光灯的shadow计算投影属性- SpotLightShadow类聚光灯的投射相机 平行光 DirectionalLight聚光灯 Sp…

12.轻量级锁原理及其实战

文章目录 轻量级锁原理及其实战1.轻量级锁的核心原理2.轻量级锁的演示2.1.轻量级锁的演示代码2.2.结果分析 3.轻量级锁的分类3.1.普通自旋锁3.2.自适应自旋锁 4.轻量级锁的膨胀 轻量级锁原理及其实战 引入轻量级锁的主要目的是在多线程环境竞争不激烈的情况下&#xff0c; 通过…

verilog中不重叠序列检测

编写一个序列检测模块&#xff0c;检测输入信号&#xff08;a&#xff09;是否满足011100序列&#xff0c; 要求以每六个输入为一组&#xff0c;不检测重复序列&#xff0c;例如第一位数据不符合&#xff0c;则不考虑后五位。一直到第七位数据即下一组信号的第一位开始检测。当…