赞
踩
MySQL的逻辑架构如下图所示:
MySQL逻辑结构中的各组件的作用说明:
Connectors:指的是不同语言中与SQL的交互。MySQL在TCP中定义了自己的应用层协议,提供了Native C API、JDBC、PHP等各语言的MySQL Connector或者ODBC,让客户端可方便与MySQL进行交互。
Management Services & Utilities:管理服务和工具组件,从备份和恢复的安全性、复制、集群、管理、配置、迁移和元数据等方面管理数据库。
Connection Pool:连接池,是为解决资源的频繁分配、释放所造成的问题而为数据库建立的一个“缓冲池”。原理:预先在缓冲池中放入一定数量的连接,当需要建立数据库连接时,只需从“缓冲池”中取出一个,使用完毕后再放回去。它的作用是进行身份验证、线程重用、连接限制、管理用户的连接、线程处理等需要缓存的需求。
SQL Interface(SQL接口):接受用户的SQL命令,并且返回用户需要查询的结果。MySQL支持DML、DDL、存储过程等多种SQL接口
Parser解析器:在解析器中对SQL语句进行语法分析、语义分析。将SQL语句分解成数据结构,并将这个结构传递到后续步骤,以后SQL语句的传递和处理就是基于这个结构。在SQL命令传到解析器的时候会被解析器验证和解析,并为其创建语法树,并根据数据字典丰富查询语法树,会验证该客户端是否具有执行该查询的权限。创建好语法树后,MySQL还会对SQL查询进行语法上的优化,进行查询重写。
Optimizer查询优化器:SQL语句在查询执行之前,会使用查询优化器对查询进行优化,得出一个最优的策略。SQL语句在语法解析之后、查询执行之前会使用查询优化器确定SQL语句的执行路径,生成一个执行计划。这个计划表明应该使用那些索引进行查询(全表检索或索引检索),表之间的连接顺序,最后会按照计划中的步骤调用存储引擎提供的方法来真正执行查询,并将结果返回给用户
Cache和Buffer:MySQL内部维持着一些Cache和Buffer,比如Query Cache用来缓存一条SELECT语句的执行结果,如果能够在其中找到对应的查询结果,那么就不必再进行查询解析、优化和执行的整个过程,直接将结果反馈给客户端。这个缓存机制是由一系列的小缓存组成,比如,表缓存、记录缓存、key缓存、权限缓存等。这个查询缓存可以在不同客户端共享。从MySQL5.7.20开始,不推荐使用查询缓存,并在MySQL8中删除。
Pluggable Storage Engines:可插拔存储引擎。
File System:数据存储在运行于裸设备的文件系统上,支持的文件类型有EXT3、EXT4 、NTFS、NFS。
Files&Logs :数据文件以及redo、undo等各种日志文件。
将MySQL的逻辑架构进行分层划分,可以分为四层,分别是:连接层,服务层,引擎层,存储层。
连接层的主要功能是:管理连接和权限验证。连接层包含本地 Socket 通信和大多数基于客户端/服务端工具实现的类似于TCP/IP 的通信。主要完成一些类似于连接处理、授权认证及相关的安全方案。
客户端访问MySQL服务器前,做的第一件事就是建立TCP连接。经过三次握手建立连接成功后,MySQL服务器对TCP传输过来的账号密码做身份认证、权限获取。
用户名或密码不对,会收到一个Access denied for user错误。
用户名密码认证通过,会从权限表查出账号拥有的权限与连接关联,之后的权限判断逻辑,都将依赖于此时读到的权限.
为了避免连接无限创建和TCP频繁创建销毁带来的资源耗尽、性能下降问题。MySQL服务器有专门的TCP连接池限制连接数,并且采用长连接模式复用TCP连接。
另外,建立TCP连接成功后,必须分配给一个线程专门与这个客户端的交互,所在该层上引入了线程池的概念,为通过安全认证接入的客户端提供线程。每一个连接从线程池中获取线程,省去了创建和销毁线程的开销。同样在该层上可以实现基于 SSL 的安全链接。服务器也会为安全接入的每个客户端验证它所具有的操作权限。
SQL Interface: SQL接口接收用户的SQL命令,并且返回用户需要查询的结果。比如SELECT ... FROM
就是调用SQL Interface。MySQL支持DML(数据操作语言)、DDL(数据定义语言)、存储过程、视图、触发器、自定义函数等多种SQL语言接口。
缓存: 查询缓存组件 MySQL内部维持着一些Cache和Buffer,比如Query Cache用来缓存一条SELECT语句的执行结果,如果能够在其中找到对应的查询结果,那么就不必再进行查询解析、优化和执行的整个过程了,直接将结果反馈给客户端。 这个缓存机制是由一系列小缓存组成的。比如表缓存,记录缓存,key缓存,权限缓存等 。 这个查询缓存可以在不同客户端之间共享 。== 从MySQL 5.7.20开始,不推荐使用查询缓存,并在 MySQL 8.0中删除== 。
Parser: 解析器 在解析器中对 SQL 语句进行语法分析、语义分析。将SQL语句分解成数据结构,并将这个结构传递到后续步骤,以后SQL语句的传递和处理就是基于这个结构的。如果在分解构成中遇到错误,那么就说明这个SQL语句是不合理的。 在SQL命令传递到解析器的时候会被解析器验证和解析,并为其创建语法树 ,并根据数据字典丰富查询语法树,会验证该客户端是否具有执行该查询的权限 。创建好语法树后,MySQL还 会对SQl查询进行语法上的优化,进行查询重写。
Optimizer: 查询优化器 SQL语句在语法解析之后、查询之前会使用查询优化器确定 SQL 语句的执行路径,生成一个 执行计划 。 这个执行计划表明应该 使用哪些索引 进行查询(全表检索还是使用索引检索),表之间的连接顺序如何,使用“ 选取-投影-连接 ”策略最后会按照执行计划中的步骤调用存储引擎提供的方法来真正的执行查询,并将查询结果返回给用户。
例如:
SELECT id,name FROM student WHERE gender = '女';
这个SELECT查询先根据WHERE语句进行选取 ,而不是将表全部查询出来以后再进行gender过滤。 然后再根据id和name进行属性投影 ,而不是将属性全部取出以后再进行过滤,将这两个查询条件 连接起来生成最终查询结果。
底层数据存取操作实现的部分,由多种存储引擎共同组成。它们负责存储和获取所有存储在MySQL中的数据,类似Linux的众多文件系统。
存储引擎层,存储引擎真正的负责了 MySQL 中数据的存储和提取,服务器通过 API 与存储引擎进行通信。不同的存储引擎具有的功能不同,这样我们可以根据自己的实际需要进行选取。
InnoDB是从MySQL 5.5.5版本开始成为了 默认存储引擎,也是最重要、使用最广泛的存储引擎。
InnoDB采用MVCC来支持高并发, 并且实现了四个标准的隔离级别。其默认级别是REPEATABLE READ (可重复读),并且通过间隙锁(next-key locking)策略防止幻读的出现。间隙锁使得InnoDB不仅仅锁定查询涉及的行,还会对索引中的间隙进行锁定,以防止幻影行的插入。
InnoDB表是基于聚簇索引建立的,聚簇索引对主键查询有很高的性能,不过它的二级索引(非主键索引)中必须包含主键列,所以如果主键列很大的话,其他的所有索引都会很大。因此如果表上的索引较多的话,主键应当尽可能小。
InnoDB内部做了很多优化,包括从磁盘读取数据时次啊用的可预测性预读,能够自动在内存中创建hash索引以加速读操作的自适应哈希索引,以及能够加速插入操作的插入缓冲区等。
MyISAM 存储引擎的特性:
InnoDB和 MyISAM 的比较
查看所有的数据库引擎的命令
show engines;
查看默认的数据库引擎的命令
show variables like '%storage_engine%';
Archive存储引擎只支持INSERT和SELECT操作,在MySQL5.1之前不支持索引。
Archive表适合日志和数据采集类应用。适合低访问量大数据等情况。
根据英文的测试结论来看,Archive表比MyISAM表要小大约75%,比支持事务处理的InnoDB表小大约83%。
Blackhole引擎没有实现任何存储机制,它会丢弃所有插入的数据,不做任何保存。但服务器会记录Blackhole表的日志,所以可以用于复制数据到备库,或者简单地记录到日志。但这种应用方式会碰到很多问题,因此并不推荐。
CSV引擎可以将普通的CSV文件作为MySQL的表来处理,但不支持索引。
CSV引擎可以作为一种数据交换的机制,非常有用。
CSV存储的数据直接可以在操作系统里,用文本编辑器,或者excel读取。
如果需要快速地访问数据,并且这些数据不会被修改,重启以后丢失也没有关系,那么使用Memory表是非常有用。Memory表至少比MyISAM表要快一个数量级。(使用专业的内存数据库更快,如redis)
Federated引擎是访问其他MySQL服务器的一个代理,尽管该引擎看起来提供了一种很好的跨服务器的灵活性,但也经常带来问题,因此默认是禁用的。
所有的数据,数据库、表的定义,表的每一行的内容,索引,都是存在文件系统上,以文件的方式存在的,并完成与存储引擎的交互。
当然有些存储引擎比如InnoDB,也支持不使用文件系统直接管理裸设备,但现代文件系统的实现使得这样做没有必要了。在文件系统之下,可以使用本地磁盘,可以使用 DAS、NAS、SAN等各种存储系统。
SQL语句在MySQL中的流程:查询缓存-》解析器-》优化器-》执行器
MySQL 客户端通过协议与 MySQL服务器建连接,发送查询语句,先检查查询缓存,如果命中,直接返回结果,否则进行语句解析,也就是说,在解析查询之前,服务器会先访问查询缓存(query cache)——它存储 SELECT 语句以及相应的查询结果集。如果某个查询结果已经位于缓存中,服务器就不会再对查询进行解析、优化、以及执行。它仅仅将缓存中的结果返回给用户即可,这将大大提高系统的性能。
但是因为查询缓存往往效率不高,所以在 MySQL8.0 之后就抛弃 了这个功能。这是为什么呢?
这是查询缓存是提前把查询结果缓存起来,这样下次不需要执行就可以直接拿到结果。需要说明的是,在 MySQL 中的查询缓存,不是缓存查询计划,而是查询对应的结果。这就意味着查询匹配的可用性大大降低 ,只有相同的查询操作才会命中查询缓存 。两个查询请求在任何字符上的不同(例如:空格、注释、 大小写),都会导致缓存不会命中。因此 MySQL 的查询缓存命中率不高 。
如果查询请求中包含某些系统函数、用户自定义变量和函数、一些系统表,如 mysql 、 information_schema、 performance_schema 数据库中的表,那这个请求就不会被缓存。以某些系统函数举例,可能同样的函数的两次调用会产生不一样的结果,比如函数 NOW ,每次调用都会产生最新的当前 时间,如果在一个查询请求中调用了这个函数,那即使查询请求的文本信息都一样,那不同时间的两次查询也应该得到不同的结果,如果在第一次查询时就缓存了,那第二次查询的时候直接使用第一次查询 的结果就是错误的!
此外,既然是缓存,那就有它缓存失效的时候 。MySQL的缓存系统会监测涉及到的每张表,只要该表的 结构或者数据被修改,如对该表使用了 INSERT 、 UPDATE 、 DELETE 、 TRUNCATE TABLE 、 ALTER TABLE 、 DROP TABLE 或 DROP DATABASE
语句,那使用该表的所有高速缓存查询都将变为无效并从高速缓存中删除!对于 更新压力大的数据库 来说,查询缓存的命中率会非常低。
解析器的主要作用是SQL的词法分析和语法分析。首先MySQL通过关键字将 SQL 语句进行解析,并生成一颗对应的“解析树”。SQL语句的分析分为词法分析与语法分析。MySQL解析器将使用 MySQL语法规则验证和解析查询;预处理器则根据一些 MySQL规则进一步检查解析数是否合法。
分析器先做“ 词法分析 ”。你输入的是由多个字符串和空格组成的一条 SQL 语句,MySQL 需要识别出里面 的字符串分别是什么,代表什么。 MySQL 从你输入的"select"这个关键字识别出来,这是一个查询语 句。它也要把字符串“T”识别成“表名 T”,把字符串“ID”识别成“列 ID”。
接着,要做“ 语法分析 ”。根据词法分析的结果,语法分析器会根据语法规则,判断你输入的这个 SQL 语句是否满足 MySQL 语法 。
例如如下SQL语句:
select username,ismale from userinfo where age>20 and level>5 and 1=1;
如果SQL语句正确,则会生成一个这样的语法树:
优化器的主要作用是生成执行计划和选择索引。 查询优化器当解析树被认为是合法的了,并且由优化器将其转化成执行计划。一条查询可以有很多种执行方式,最后都返回相同的结果。优化器的作用就是找到这其中最好的执行计划。比如是根据全表检索 ,还是根据索引检索等。
在查询优化器中,可以分为逻辑查询优化阶段和物理查询优化阶段。
逻辑查询优化:逻辑查询优化就是通过改变SQL语句的内容来使得SQL查询更加高效,同时为物理查询优化提供更多的候选执行计划。通常采用的方式是对SQL语句进行等价变换,对查询进行重写,而查询重写的数学基础就是关系代数。对条件表达式进行等价谓词重写、条件简化,对视图进行重写,对子查询进行优化,对连接语句进行了外连接消除、嵌套连接消除等。
物理查询优化:物理查询优化基于关系代数进行的查询重写,而关系代数的每一步都对应着物理计算。在这个阶段里,对于单表和多表的连接操作,需要高效地使用索引,提升查询效率。
执行器的主要功能是操作存储引擎和返回结果。执行器在执行之前需要判断该用户是否具备权限,如果没有就会返回权限错误。如果有权限,就打开表继续执行,执行器会根据表的引擎定义,调用存储引擎API对表进行读写。存储引擎API只是抽象接口,下面还有个存储引擎层。最终从存储引擎层返回结果给到客户端。
一条SQL语句的模板如下所示:
随着 MySQL版本的更新换代,其优化器也在不断的升级,优化器会分析不同执行顺序产生的性能消耗不同而动态调整执行顺序。下面是经常出现的查询顺序:
(8) SELECT (9)DISTINCT<select_list>
(1) FROM <left_table>
(3) <join_type> JOIN <right_table>
(2) ON <join_condition>
(4) WHERE <where_condition>
(5) GROUP BY <group_by_list>
(6) WITH {CUBE|ROLLUP}
(7) HAVING <having_condition>
(10) ORDER BY <order_by_list>
(11) LIMIT <limit_number>
一共有十一个步骤,最先执行的是FROM操作,最后执行的是LIMIT操作。每个操作都会产生一个虚拟表,该虚拟表作为一个处理的输入,具体执行顺序如下:
(1) FROM:对FROM子句中的左表<left_table>和右表<right_table>执行笛卡儿积,产生虚拟表VT1;
(2) ON: 对虚拟表VT1进行ON筛选,只有那些符合<join_condition>的行才被插入虚拟表VT2;
(3) JOIN: 如果指定了OUTER JOIN(如LEFT OUTER JOIN、RIGHT OUTER JOIN),那么保留表中未匹配的行作为外部行添加到虚拟表VT2,产生虚拟表VT3。如果FROM子句包含两个以上的表,则对上一个连接生成的结果表VT3和下一个表重复执行步骤1~步骤3,直到处理完所有的表;
(4) WHERE: 对虚拟表VT3应用WHERE过滤条件,只有符合<where_condition>的记录才会被插入虚拟表VT4;
(5) GROUP By: 根据GROUP BY子句中的列,对VT4中的记录进行分组操作,产生VT5;
(6) CUBE|ROllUP: 对VT5进行CUBE或ROLLUP操作,产生表VT6;
(7) HAVING: 对虚拟表VT6应用HAVING过滤器,只有符合<having_condition>的记录才会被插入到VT7;
(8) SELECT: 执行SELECT操作,选择指定的列,插入到虚拟表VT8中;
(9) DISTINCT: 去除重复数据,产生虚拟表VT9;
(10) ORDER BY: 将虚拟表VT9中的记录按照<order_by_list>进行排序操作,产生虚拟表VT10;
(11) LIMIT: 取出指定行的记录,产生虚拟表VT11,并返回给查询用户
(1)修改配置文件/etc/my.cnf,新增一行:query_cache_type=1
重启MySQL服务
systemctl restart mysqld
参考:
https://blog.csdn.net/weixin_50616848/article/details/125583895
查看 profile 是否开启的命令
show variables like '%profiling%';
profiling为OFF,则为没有开启。如果没有开启,可以执行以下命令开启!
或者使用以下命令:
select @@profiling;
开启profile功能的命令
set profiling=1;
执行 show profiles 命令,可以查看最近的几次查询和执行的命令。
show profiles;
show profile命令说明
show profile [type,type,..] for query n limit row_count [offset offset]
type的可选值:
根据 Query_ID,可以进一步执行 show profile cpu,block io for query Query_id 来查看 sql 的具体执行步骤。
select * from mydb.user where id=1;
根据 Query_ID,查看SQL语句的具体执行步骤。
show profiles;
show profile cpu,block io for query 4;
当开启了查询缓存,并且命中缓存(SQL语句必须一致才可以缓存命中,如果对表进行了insert,update,delete等操作,缓存则会失效),SQL语句的大致执行步骤如下所示:
InnoDB存储引擎是以页为单位来管理存储空间的,我们进行的增删改查操作其实本质上都是在访问页面。而磁盘I/O需要消耗的时间很多,而在内存中进行操作,效率则会高很多,为了能让数据表或者索引数据被我们所用,DBMS会申请占用内存来作为数据缓冲池,在真正访问页面之前,需要把磁盘上的页缓存到内存中的buffer pool之后才能访问,从而让磁盘活动最小化,减少与磁盘直接进行I/O的时间。
缓冲池用来存储各种数据的缓存,如下图所示:
InnoDB缓冲池包括了数据页,索引页,插入缓存,锁信息,自适应Hash和数据字典信息等。
缓存原则
位置 * 频次这个原则,可以帮我们对I/O访问效率进行优化。首先位置决定效率,提供缓冲池就是为了在内存中可以直接访问数据。其次,频次决定优先级顺序。因为缓冲池的大小是有限的,会优先对使用频次高的热数据进行加载
缓冲池的预读特性
缓冲池的作用就是提升I/O效率,而我们进行读取数据的时候存在一个局部性原理,也就是说我们使用了一些数据,大概率还会使用它周围的一些数据,因此采用预读机制提前加载,可以减少未来可能的磁盘I/O操作
缓冲池管理器会尽量将使用的数据保存起来,在数据库进行页面操作读操作的时候,首先会判断该页是否存在缓冲池中,如果存在就直接读取,如果不存在,就会通过内存或磁盘将页面放到缓冲池中再进行读取。
如果我们执行SQL语句的时候更新了缓冲池中的数据,那么这些数据会马上同步到磁盘吗?
实际上,当我们对数据库中的记录进行修改的时候,首先会修改缓冲池中页里面的记录信息,然后数据库会以一定的频率刷新到磁盘。缓冲池会采用一种叫做checkpoint的机制将数据回写到磁盘上。
当缓冲池不够用时,需要释放掉一些不常用的页,此时就可以强行采用checkpoint的方式,将不常用的脏页回写到磁盘上,然后再从缓冲池中将这些页释放掉。
通过innodb_buffer_pool_size变量来查看缓冲池的大小
show variables like 'innodb_buffer_pool_size';
修改缓冲池的大小的命令
set global innodb_buffer_pool_size =
Buffer Pool的本质是InnoDB向操作系统申请的一块连续的内存空间,在多线程环境下,访问Buffer pool中的数据都需要加锁处理。在Buffer pool特别大而且多线程并发访问特别高的情况下,单一的Buffer Pool可能影响处理的请求速度。所以在Buffer Pool特别大的时候,我们可以把它们差分成若干个小的Buffer Poll,它们独立出去,独立申请内存空间,独立的管理各种链表,所以在多线程并发访问时并不会相互影响,从而提高并发处理能力。
我们可以在服务器启动的时候设置innodb_buffer_pool_instances的值来修改Buffer Pool实例的个数。
[server]
#创建2个Buffer Pool实例
innodb_buffer_pool_instance = 2;
查看缓冲池个数的命令
show variables like 'innodb_buffer_pool_instances';
默认情况下的Buffer Pool只有1。每个Buffer Pool实例占用的内存大小等于innodb_buffer_pool_size/innodb_buffer_pool_instance
。因为管理各个Buffer Pool也是需要性能开销的,所以并不是Buffer Pool实例创建越多越好。InnoDB规定:当innodb_buffer_pool_size的值小于1G的时候设置多个实例是无效的,InnoDB会默认把innodb_buffer_pool_instance的值修改为1.当innodb_buffer_pool_size的值大于或等于1G的时候,推荐设置多个Buffer Pool实例。
参考:
https://blog.csdn.net/qq_27495855/article/details/124326854
https://blog.csdn.net/weixin_51538712/article/details/124586938
https://blog.csdn.net/qq_38327769/article/details/12421975
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。