详解MySQL的逻辑架构和SQL语句执行流程

文章目录

    • 1. 逻辑架构
      • 1.1 连接层
      • 1.2 服务层
      • 1.3 引擎层
        • 1.3.1 InnoDB 存储引擎
        • 1.3.2 MyISAM 存储引擎
        • 1.3.3 其他存储引擎
      • 1.4 存储层
    • 3. SQL语句执行流程
      • 3.1 查询缓存
      • 3.2 解析器
      • 3.3 优化器
      • 3.4 执行器
    • 4. SQL语句的执行顺序
    • 5. profile功能
      • 5.1 开启查询缓存
      • 5.2 开启profile
      • 5.3 使用profile
    • 6. 数据库缓冲池
      • 6.1 缓存池如何读取数据
      • 6.2 操作缓冲池

1. 逻辑架构

MySQL的逻辑架构如下图所示:
详解MySQL的逻辑架构和SQL语句执行流程_第1张图片

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的逻辑架构进行分层划分,可以分为四层,分别是:连接层,服务层,引擎层,存储层。

1.1 连接层

连接层的主要功能是:管理连接和权限验证。连接层包含本地 Socket 通信和大多数基于客户端/服务端工具实现的类似于TCP/IP 的通信。主要完成一些类似于连接处理、授权认证及相关的安全方案。

客户端访问MySQL服务器前,做的第一件事就是建立TCP连接。经过三次握手建立连接成功后,MySQL服务器对TCP传输过来的账号密码做身份认证、权限获取。

  • 用户名或密码不对,会收到一个Access denied for user错误。

  • 用户名密码认证通过,会从权限表查出账号拥有的权限与连接关联,之后的权限判断逻辑,都将依赖于此时读到的权限.

为了避免连接无限创建和TCP频繁创建销毁带来的资源耗尽、性能下降问题。MySQL服务器有专门的TCP连接池限制连接数,并且采用长连接模式复用TCP连接。

另外,建立TCP连接成功后,必须分配给一个线程专门与这个客户端的交互,所在该层上引入了线程池的概念,为通过安全认证接入的客户端提供线程。每一个连接从线程池中获取线程,省去了创建和销毁线程的开销。同样在该层上可以实现基于 SSL 的安全链接。服务器也会为安全接入的每个客户端验证它所具有的操作权限。

1.2 服务层

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进行属性投影 ,而不是将属性全部取出以后再进行过滤,将这两个查询条件 连接起来生成最终查询结果。

1.3 引擎层

底层数据存取操作实现的部分,由多种存储引擎共同组成。它们负责存储和获取所有存储在MySQL中的数据,类似Linux的众多文件系统。

存储引擎层,存储引擎真正的负责了 MySQL 中数据的存储和提取,服务器通过 API 与存储引擎进行通信。不同的存储引擎具有的功能不同,这样我们可以根据自己的实际需要进行选取。

1.3.1 InnoDB 存储引擎

InnoDB是从MySQL 5.5.5版本开始成为了 默认存储引擎,也是最重要、使用最广泛的存储引擎。

InnoDB采用MVCC来支持高并发, 并且实现了四个标准的隔离级别。其默认级别是REPEATABLE READ (可重复读),并且通过间隙锁(next-key locking)策略防止幻读的出现。间隙锁使得InnoDB不仅仅锁定查询涉及的行,还会对索引中的间隙进行锁定,以防止幻影行的插入。

InnoDB表是基于聚簇索引建立的,聚簇索引对主键查询有很高的性能,不过它的二级索引(非主键索引)中必须包含主键列,所以如果主键列很大的话,其他的所有索引都会很大。因此如果表上的索引较多的话,主键应当尽可能小。

InnoDB内部做了很多优化,包括从磁盘读取数据时次啊用的可预测性预读,能够自动在内存中创建hash索引以加速读操作的自适应哈希索引,以及能够加速插入操作的插入缓冲区等。

1.3.2 MyISAM 存储引擎

MyISAM 存储引擎的特性:

  • 不支持事务、不具备AICD特性(原子性、一致性、分离性、可持续性)
  • 表级别锁定形式(更新数据时锁定整个表、这样虽然可以让锁定的实现成本很小但是同时大大降低了其并发的性能)
  • 读写相互阻塞(不仅会在写入的时候阻塞读取、还会在读取的时候阻塞写入、但是读取不会阻塞读取)
    只会缓存索引(myisam通过key_buffer_size来设置缓存索引,提高访问性能较少磁盘IO的压力、但是只缓存索引、不缓存数据)
  • 读取速度快、占用资源比较少
  • 不支持外键约束、只支持全文检索
  • MyISAM表支持数据压缩

InnoDB和 MyISAM 的比较
详解MySQL的逻辑架构和SQL语句执行流程_第2张图片
详解MySQL的逻辑架构和SQL语句执行流程_第3张图片

查看所有的数据库引擎的命令

show engines;

详解MySQL的逻辑架构和SQL语句执行流程_第4张图片
查看默认的数据库引擎的命令

show variables like '%storage_engine%';

详解MySQL的逻辑架构和SQL语句执行流程_第5张图片

1.3.3 其他存储引擎

  • Archive引擎

Archive存储引擎只支持INSERT和SELECT操作,在MySQL5.1之前不支持索引。

Archive表适合日志和数据采集类应用。适合低访问量大数据等情况。
根据英文的测试结论来看,Archive表比MyISAM表要小大约75%,比支持事务处理的InnoDB表小大约83%。

  • Blackhole引擎

Blackhole引擎没有实现任何存储机制,它会丢弃所有插入的数据,不做任何保存。但服务器会记录Blackhole表的日志,所以可以用于复制数据到备库,或者简单地记录到日志。但这种应用方式会碰到很多问题,因此并不推荐。

  • CSV引擎

CSV引擎可以将普通的CSV文件作为MySQL的表来处理,但不支持索引。
CSV引擎可以作为一种数据交换的机制,非常有用。

CSV存储的数据直接可以在操作系统里,用文本编辑器,或者excel读取。

  • Memory引擎

如果需要快速地访问数据,并且这些数据不会被修改,重启以后丢失也没有关系,那么使用Memory表是非常有用。Memory表至少比MyISAM表要快一个数量级。(使用专业的内存数据库更快,如redis)

  • Federated引擎

Federated引擎是访问其他MySQL服务器的一个代理,尽管该引擎看起来提供了一种很好的跨服务器的灵活性,但也经常带来问题,因此默认是禁用的。

1.4 存储层

所有的数据,数据库、表的定义,表的每一行的内容,索引,都是存在文件系统上,以文件的方式存在的,并完成与存储引擎的交互。

当然有些存储引擎比如InnoDB,也支持不使用文件系统直接管理裸设备,但现代文件系统的实现使得这样做没有必要了。在文件系统之下,可以使用本地磁盘,可以使用 DAS、NAS、SAN等各种存储系统。

3. SQL语句执行流程

SQL语句在MySQL中的流程:查询缓存-》解析器-》优化器-》执行器

详解MySQL的逻辑架构和SQL语句执行流程_第6张图片

3.1 查询缓存

MySQL 客户端通过协议与 MySQL服务器建连接,发送查询语句,先检查查询缓存,如果命中,直接返回结果,否则进行语句解析,也就是说,在解析查询之前,服务器会先访问查询缓存(query cache)——它存储 SELECT 语句以及相应的查询结果集。如果某个查询结果已经位于缓存中,服务器就不会再对查询进行解析、优化、以及执行。它仅仅将缓存中的结果返回给用户即可,这将大大提高系统的性能。

但是因为查询缓存往往效率不高,所以在 MySQL8.0 之后就抛弃 了这个功能。这是为什么呢?

  1. 这是查询缓存是提前把查询结果缓存起来,这样下次不需要执行就可以直接拿到结果。需要说明的是,在 MySQL 中的查询缓存,不是缓存查询计划,而是查询对应的结果。这就意味着查询匹配的可用性大大降低 ,只有相同的查询操作才会命中查询缓存 。两个查询请求在任何字符上的不同(例如:空格、注释、 大小写),都会导致缓存不会命中。因此 MySQL 的查询缓存命中率不高 。

  2. 如果查询请求中包含某些系统函数、用户自定义变量和函数、一些系统表,如 mysql 、 information_schema、 performance_schema 数据库中的表,那这个请求就不会被缓存。以某些系统函数举例,可能同样的函数的两次调用会产生不一样的结果,比如函数 NOW ,每次调用都会产生最新的当前 时间,如果在一个查询请求中调用了这个函数,那即使查询请求的文本信息都一样,那不同时间的两次查询也应该得到不同的结果,如果在第一次查询时就缓存了,那第二次查询的时候直接使用第一次查询 的结果就是错误的!

  3. 此外,既然是缓存,那就有它缓存失效的时候 。MySQL的缓存系统会监测涉及到的每张表,只要该表的 结构或者数据被修改,如对该表使用了 INSERT 、 UPDATE 、 DELETE 、 TRUNCATE TABLE 、 ALTER TABLE 、 DROP TABLE 或 DROP DATABASE 语句,那使用该表的所有高速缓存查询都将变为无效并从高速缓存中删除!对于 更新压力大的数据库 来说,查询缓存的命中率会非常低。

3.2 解析器

解析器的主要作用是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语句正确,则会生成一个这样的语法树:
详解MySQL的逻辑架构和SQL语句执行流程_第7张图片

3.3 优化器

优化器的主要作用是生成执行计划和选择索引。 查询优化器当解析树被认为是合法的了,并且由优化器将其转化成执行计划。一条查询可以有很多种执行方式,最后都返回相同的结果。优化器的作用就是找到这其中最好的执行计划。比如是根据全表检索 ,还是根据索引检索等。

在查询优化器中,可以分为逻辑查询优化阶段和物理查询优化阶段。

  • 逻辑查询优化:逻辑查询优化就是通过改变SQL语句的内容来使得SQL查询更加高效,同时为物理查询优化提供更多的候选执行计划。通常采用的方式是对SQL语句进行等价变换,对查询进行重写,而查询重写的数学基础就是关系代数。对条件表达式进行等价谓词重写、条件简化,对视图进行重写,对子查询进行优化,对连接语句进行了外连接消除、嵌套连接消除等。

  • 物理查询优化:物理查询优化基于关系代数进行的查询重写,而关系代数的每一步都对应着物理计算。在这个阶段里,对于单表和多表的连接操作,需要高效地使用索引,提升查询效率。

3.4 执行器

执行器的主要功能是操作存储引擎和返回结果。执行器在执行之前需要判断该用户是否具备权限,如果没有就会返回权限错误。如果有权限,就打开表继续执行,执行器会根据表的引擎定义,调用存储引擎API对表进行读写。存储引擎API只是抽象接口,下面还有个存储引擎层。最终从存储引擎层返回结果给到客户端。

4. SQL语句的执行顺序

一条SQL语句的模板如下所示:
详解MySQL的逻辑架构和SQL语句执行流程_第8张图片

随着 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子句中的左表和右表执行笛卡儿积,产生虚拟表VT1;
(2) ON: 对虚拟表VT1进行ON筛选,只有那些符合的行才被插入虚拟表VT2;
(3) JOIN: 如果指定了OUTER JOIN(如LEFT OUTER JOIN、RIGHT OUTER JOIN),那么保留表中未匹配的行作为外部行添加到虚拟表VT2,产生虚拟表VT3。如果FROM子句包含两个以上的表,则对上一个连接生成的结果表VT3和下一个表重复执行步骤1~步骤3,直到处理完所有的表;
(4) WHERE: 对虚拟表VT3应用WHERE过滤条件,只有符合的记录才会被插入虚拟表VT4;
(5) GROUP By: 根据GROUP BY子句中的列,对VT4中的记录进行分组操作,产生VT5;
(6) CUBE|ROllUP: 对VT5进行CUBE或ROLLUP操作,产生表VT6;
(7) HAVING: 对虚拟表VT6应用HAVING过滤器,只有符合的记录才会被插入到VT7;
(8) SELECT: 执行SELECT操作,选择指定的列,插入到虚拟表VT8中;
(9) DISTINCT: 去除重复数据,产生虚拟表VT9;
(10) ORDER BY: 将虚拟表VT9中的记录按照进行排序操作,产生虚拟表VT10;
(11) LIMIT: 取出指定行的记录,产生虚拟表VT11,并返回给查询用户

5. profile功能

5.1 开启查询缓存

(1)修改配置文件/etc/my.cnf,新增一行:query_cache_type=1

详解MySQL的逻辑架构和SQL语句执行流程_第9张图片
重启MySQL服务

systemctl restart mysqld

参考:
https://blog.csdn.net/weixin_50616848/article/details/125583895

5.2 开启profile

查看 profile 是否开启的命令

show variables like '%profiling%';

详解MySQL的逻辑架构和SQL语句执行流程_第10张图片
profiling为OFF,则为没有开启。如果没有开启,可以执行以下命令开启!

或者使用以下命令:

select @@profiling;

开启profile功能的命令

 set profiling=1;

详解MySQL的逻辑架构和SQL语句执行流程_第11张图片

5.3 使用profile

执行 show profiles 命令,可以查看最近的几次查询和执行的命令。

 show profiles;

详解MySQL的逻辑架构和SQL语句执行流程_第12张图片
show profile命令说明

show profile [type,type,..] for query n limit row_count [offset offset]

type的可选值:

  • ALL --显示所有参数的开销信息
  • BLOCK IO --显示IO相关开销
  • CONTEXT SWITCHES --上下文切换相关开销
  • CPU --显示CPU相关开销信息
  • IPC --显示发送和接收相关开销信息
  • MEMORY --显示内存相关开销信息
  • PAGE FAULTS --显示页面错误相关开销信息
  • SOURCE --显示和Source_function,Source_file,Source_line相关开销信息
  • SWAPS --显示交换次数相关开销信息

根据 Query_ID,可以进一步执行 show profile cpu,block io for query Query_id 来查看 sql 的具体执行步骤。

 select * from mydb.user where id=1;

详解MySQL的逻辑架构和SQL语句执行流程_第13张图片

根据 Query_ID,查看SQL语句的具体执行步骤。

 show profiles;
show profile cpu,block io for query 4;

详解MySQL的逻辑架构和SQL语句执行流程_第14张图片

当开启了查询缓存,并且命中缓存(SQL语句必须一致才可以缓存命中,如果对表进行了insert,update,delete等操作,缓存则会失效),SQL语句的大致执行步骤如下所示:
详解MySQL的逻辑架构和SQL语句执行流程_第15张图片

6. 数据库缓冲池

InnoDB存储引擎是以页为单位来管理存储空间的,我们进行的增删改查操作其实本质上都是在访问页面。而磁盘I/O需要消耗的时间很多,而在内存中进行操作,效率则会高很多,为了能让数据表或者索引数据被我们所用,DBMS会申请占用内存来作为数据缓冲池,在真正访问页面之前,需要把磁盘上的页缓存到内存中的buffer pool之后才能访问,从而让磁盘活动最小化,减少与磁盘直接进行I/O的时间。

缓冲池用来存储各种数据的缓存,如下图所示:
详解MySQL的逻辑架构和SQL语句执行流程_第16张图片
InnoDB缓冲池包括了数据页,索引页,插入缓存,锁信息,自适应Hash和数据字典信息等。

  • 缓存原则
    位置 * 频次这个原则,可以帮我们对I/O访问效率进行优化。首先位置决定效率,提供缓冲池就是为了在内存中可以直接访问数据。其次,频次决定优先级顺序。因为缓冲池的大小是有限的,会优先对使用频次高的热数据进行加载

  • 缓冲池的预读特性
    缓冲池的作用就是提升I/O效率,而我们进行读取数据的时候存在一个局部性原理,也就是说我们使用了一些数据,大概率还会使用它周围的一些数据,因此采用预读机制提前加载,可以减少未来可能的磁盘I/O操作

6.1 缓存池如何读取数据

缓冲池管理器会尽量将使用的数据保存起来,在数据库进行页面操作读操作的时候,首先会判断该页是否存在缓冲池中,如果存在就直接读取,如果不存在,就会通过内存或磁盘将页面放到缓冲池中再进行读取。

详解MySQL的逻辑架构和SQL语句执行流程_第17张图片

如果我们执行SQL语句的时候更新了缓冲池中的数据,那么这些数据会马上同步到磁盘吗?
实际上,当我们对数据库中的记录进行修改的时候,首先会修改缓冲池中页里面的记录信息,然后数据库会以一定的频率刷新到磁盘。缓冲池会采用一种叫做checkpoint的机制将数据回写到磁盘上。
当缓冲池不够用时,需要释放掉一些不常用的页,此时就可以强行采用checkpoint的方式,将不常用的脏页回写到磁盘上,然后再从缓冲池中将这些页释放掉。

6.2 操作缓冲池

通过innodb_buffer_pool_size变量来查看缓冲池的大小

show variables like 'innodb_buffer_pool_size';

详解MySQL的逻辑架构和SQL语句执行流程_第18张图片

修改缓冲池的大小的命令

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';

详解MySQL的逻辑架构和SQL语句执行流程_第19张图片
默认情况下的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

你可能感兴趣的:(MySQL,mysql,sql,架构)